go-libp2p-example-chat学习

2023-12-13 06:36
文章标签 go 学习 example chat libp2p

本文主要是介绍go-libp2p-example-chat学习,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.案例下载

https://github.com/libp2p/go-libp2p/tree/master/examples

2.chat案例

这段代码是一个简单的基于libp2p的P2P聊天应用程序的示例。它允许两个节点通过P2P连接进行聊天。前提是:

  1. 两者都有私有IP地址(同一网络)。
  2. 至少其中一个具有公共IP地址。

假设如果’A’和’B’在不同的网络上,主机’A’可能有或可能没有公共IP地址,但主机’B’一定有一个公共IP地址。

//在一个命令行输入
`./chat -sp 3001` 
//在另一个命令后输入一下代码,<MULTIADDR_B>`是前一个与前节点通信的标识
`./chat -d <MULTIADDR_B>` 

运行后效果如下:
在这里插入图片描述

3.源码分析

3.1 main函数

func main() {// 创建一个上下文和取消函数以进行 graceful shutdownctx, cancel := context.WithCancel(context.Background())defer cancel()// 定义命令行参数sourcePort := flag.Int("sp", 0, "Source port number")dest := flag.String("d", "", "Destination multiaddr string")help := flag.Bool("help", false, "Display help")debug := flag.Bool("debug", false, "Debug generates the same node ID on every execution")// 解析命令行参数flag.Parse()// 如果传递了-help参数,显示帮助信息并退出if *help {fmt.Printf("This program demonstrates a simple p2p chat application using libp2p\n\n")fmt.Println("Usage: Run './chat -sp <SOURCE_PORT>' where <SOURCE_PORT> can be any port number.")fmt.Println("Now run './chat -d <MULTIADDR>' where <MULTIADDR> is multiaddress of previous listener host.")os.Exit(0)}// 如果启用调试模式,则使用常量随机源生成对等方ID。仅用于调试,默认情况下关闭。否则,它将使用 rand.Reader。var r io.Readerif *debug {// 使用端口号作为随机源。// 如果使用相同的端口号,这将在多次执行中始终生成相同的主机ID。// 在生产代码中永远不要这样做。r = mrand.New(mrand.NewSource(int64(*sourcePort)))} else {r = rand.Reader}// 创建libp2p主机h, err := makeHost(*sourcePort, r)if err != nil {log.Println(err)return}// 如果未指定目标地址,则作为监听者启动节点if *dest == "" {startPeer(ctx, h, handleStream)} else {// 如果指定了目标地址,表明这是一个主动连接的节点。需要创建线程以读取和写入数据rw, err := startPeerAndConnect(ctx, h, *dest)if err != nil {log.Println(err)return}// 创建线程以读取和写入数据go writeData(rw)go readData(rw)}// 永久等待select {}
}

3.2 makeHost

// makeHost函数用于创建libp2p主机
func makeHost(port int, randomness io.Reader) (host.Host, error) {// 为此主机创建一个新的RSA密钥对,返回私钥、公钥、错误信息prvKey, _, err := crypto.GenerateKeyPairWithReader(crypto.RSA, 2048, randomness)if err != nil {log.Println(err)return nil, err}// 0.0.0.0 将监听任何接口设备。sourceMultiAddr, _ := multiaddr.NewMultiaddr(fmt.Sprintf("/ip4/0.0.0.0/tcp/%d", port))// libp2p.New 用于构建一个新的libp2p主机。// 这里可以添加其他选项。return libp2p.New(libp2p.ListenAddrs(sourceMultiAddr),  // 设置主机监听的地址libp2p.Identity(prvKey),               // 设置主机的身份,即密钥对)
}

3.3 startPeer

// startPeer函数用于启动作为监听者的节点
func startPeer(ctx context.Context, h host.Host, streamHandler network.StreamHandler) {// 设置一个函数作为流处理器。// 当节点连接时,此函数被调用,并启动一个带有该协议的流。// 仅适用于接收方,这里使用的streamHandler是代码中的handleStreamh.SetStreamHandler("/chat/1.0.0", streamHandler)// 让我们从我们的监听多地址中获取实际的TCP端口,以防我们使用0(默认值;随机可用端口)。var port stringfor _, la := range h.Network().ListenAddresses() {if p, err := la.ValueForProtocol(multiaddr.P_TCP); err == nil {port = pbreak}}if port == "" {log.Println("无法找到实际的本地端口")return}log.Printf("在另一个控制台中运行 './chat -d /ip4/127.0.0.1/tcp/%v/p2p/%s'\n", port, h.ID())log.Println("您也可以用公共IP替换 127.0.0.1。")log.Println("等待传入连接")log.Println()
}

3.4 startPeerAndConnect

// startPeerAndConnect函数用于启动作为主动连接者的节点并连接到目标地址
func startPeerAndConnect(ctx context.Context, h host.Host, destination string) (*bufio.ReadWriter, error) {log.Println("此节点的多地址:")for _, la := range h.Addrs() {log.Printf(" - %v\n", la)}log.Println()// 将目标地址转换为multiaddr。maddr, err := multiaddr.NewMultiaddr(destination)if err != nil {log.Println(err)return nil, err}// 从multiaddr中提取对等方ID。info, err := peer.AddrInfoFromP2pAddr(maddr)if err != nil {log.Println(err)return nil, err}// 重要:在peerstore中添加目标对等方的对等多地址。// 这将在libp2p的连接和流创建过程中使用。// info.ID是一个peer的唯一标识,根据ID可以找到对应的多地址h.Peerstore().AddAddrs(info.ID, info.Addrs, peerstore.PermanentAddrTTL)// 与目标建立流。// 使用 'peerId' 从peerstore中获取目标对等方的多地址。s, err := h.NewStream(context.Background(), info.ID, "/chat/1.0.0")if err != nil {log.Println(err)return nil, err}log.Println("已建立到目标的连接")// 创建一个带有缓冲的流,以使读取和写入操作不会阻塞。rw := bufio.NewReadWriter(bufio.NewReader(s), bufio.NewWriter(s))return rw, nil
}

3.5 handleStream

func handleStream(s network.Stream) {log.Println("Got a new stream!")// 创建一个不堵塞的读写缓冲流rw := bufio.NewReadWriter(bufio.NewReader(s), bufio.NewWriter(s))go readData(rw)go writeData(rw)// 流 's' 将保持打开状态,直到您关闭它(或另一侧关闭它)。
}

3.6 readData和writeData

// readData函数用于从读写器中读取数据并在控制台上显示
func readData(rw *bufio.ReadWriter) {for {// 从读写器中读取字符串,直到遇到换行符 '\n'str, _ := rw.ReadString('\n')// 如果字符串为空,则退出循环if str == "" {return}// 如果字符串不是空行 '\n'if str != "\n" {// 控制台显示绿色文本:	\x1b[32m// 重置控制台文本颜色:	\x1b[0mfmt.Printf("\x1b[32m%s\x1b[0m> ", str)}}
}// writeData函数用于从标准输入读取数据并将其写入到读写器中
func writeData(rw *bufio.ReadWriter) {// 创建一个用于读取标准输入的读取器stdReader := bufio.NewReader(os.Stdin)for {// 提示符fmt.Print("> ")// 从标准输入读取数据,直到遇到换行符 '\n'sendData, err := stdReader.ReadString('\n')if err != nil {log.Println(err)return}// 将数据写入读写器,并刷新缓冲rw.WriteString(fmt.Sprintf("%s\n", sendData))rw.Flush()}
}

这篇关于go-libp2p-example-chat学习的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/487446

相关文章

Go学习记录之runtime包深入解析

《Go学习记录之runtime包深入解析》Go语言runtime包管理运行时环境,涵盖goroutine调度、内存分配、垃圾回收、类型信息等核心功能,:本文主要介绍Go学习记录之runtime包的... 目录前言:一、runtime包内容学习1、作用:① Goroutine和并发控制:② 垃圾回收:③ 栈和

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

Go语言如何判断两张图片的相似度

《Go语言如何判断两张图片的相似度》这篇文章主要为大家详细介绍了Go语言如何中实现判断两张图片的相似度的两种方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 在介绍技术细节前,我们先来看看图片对比在哪些场景下可以用得到:图片去重:自动删除重复图片,为存储空间"瘦身"。想象你是一个

Go语言中Recover机制的使用

《Go语言中Recover机制的使用》Go语言的recover机制通过defer函数捕获panic,实现异常恢复与程序稳定性,具有一定的参考价值,感兴趣的可以了解一下... 目录引言Recover 的基本概念基本代码示例简单的 Recover 示例嵌套函数中的 Recover项目场景中的应用Web 服务器中

Android学习总结之Java和kotlin区别超详细分析

《Android学习总结之Java和kotlin区别超详细分析》Java和Kotlin都是用于Android开发的编程语言,它们各自具有独特的特点和优势,:本文主要介绍Android学习总结之Ja... 目录一、空安全机制真题 1:Kotlin 如何解决 Java 的 NullPointerExceptio

Go语言中使用JWT进行身份验证的几种方式

《Go语言中使用JWT进行身份验证的几种方式》本文主要介绍了Go语言中使用JWT进行身份验证的几种方式,包括dgrijalva/jwt-go、golang-jwt/jwt、lestrrat-go/jw... 目录简介1. github.com/dgrijalva/jwt-go安装:使用示例:解释:2. gi

go rate 原生标准限速库的使用

《gorate原生标准限速库的使用》本文主要介绍了Go标准库golang.org/x/time/rate实现限流,采用令牌桶算法控制请求速率,提供Allow/Reserve/Wait方法,具有一定... 目录介绍安装API介绍rate.NewLimiter:创建限流器limiter.Allow():请求是否

Go 语言中的 Struct Tag 的用法详解

《Go语言中的StructTag的用法详解》在Go语言中,结构体字段标签(StructTag)是一种用于给字段添加元信息(metadata)的机制,常用于序列化(如JSON、XML)、ORM映... 目录一、结构体标签的基本语法二、json:"token"的具体含义三、常见的标签格式变体四、使用示例五、使用

Ubuntu上手动安装Go环境并解决“可执行文件格式错误”问题

《Ubuntu上手动安装Go环境并解决“可执行文件格式错误”问题》:本文主要介绍Ubuntu上手动安装Go环境并解决“可执行文件格式错误”问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未... 目录一、前言二、系统架构检测三、卸载旧版 Go四、下载并安装正确版本五、配置环境变量六、验证安装七、常见

Go语言使用slices包轻松实现排序功能

《Go语言使用slices包轻松实现排序功能》在Go语言开发中,对数据进行排序是常见的需求,Go1.18版本引入的slices包提供了简洁高效的排序解决方案,支持内置类型和用户自定义类型的排序操作,本... 目录一、内置类型排序:字符串与整数的应用1. 字符串切片排序2. 整数切片排序二、检查切片排序状态: