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语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

Go语言中json操作的实现

《Go语言中json操作的实现》本文主要介绍了Go语言中的json操作的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录 一、jsOChina编程N 与 Go 类型对应关系️ 二、基本操作:编码与解码 三、结构体标签(Struc

使用Go调用第三方API的方法详解

《使用Go调用第三方API的方法详解》在现代应用开发中,调用第三方API是非常常见的场景,比如获取天气预报、翻译文本、发送短信等,Go作为一门高效并发的编程语言,拥有强大的标准库和丰富的第三方库,可以... 目录引言一、准备工作二、案例1:调用天气查询 API1. 注册并获取 API Key2. 代码实现3

基于Go语言开发一个 IP 归属地查询接口工具

《基于Go语言开发一个IP归属地查询接口工具》在日常开发中,IP地址归属地查询是一个常见需求,本文将带大家使用Go语言快速开发一个IP归属地查询接口服务,有需要的小伙伴可以了解下... 目录功能目标技术栈项目结构核心代码(main.go)使用方法扩展功能总结在日常开发中,IP 地址归属地查询是一个常见需求:

GO语言短变量声明的实现示例

《GO语言短变量声明的实现示例》在Go语言中,短变量声明是一种简洁的变量声明方式,使用:=运算符,可以自动推断变量类型,下面就来具体介绍一下如何使用,感兴趣的可以了解一下... 目录基本语法功能特点与var的区别适用场景注意事项基本语法variableName := value功能特点1、自动类型推

GO语言中函数命名返回值的使用

《GO语言中函数命名返回值的使用》在Go语言中,函数可以为其返回值指定名称,这被称为命名返回值或命名返回参数,这种特性可以使代码更清晰,特别是在返回多个值时,感兴趣的可以了解一下... 目录基本语法函数命名返回特点代码示例命名特点基本语法func functionName(parameters) (nam

Go之errors.New和fmt.Errorf 的区别小结

《Go之errors.New和fmt.Errorf的区别小结》本文主要介绍了Go之errors.New和fmt.Errorf的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考... 目录error的基本用法1. 获取错误信息2. 在条件判断中使用基本区别1.函数签名2.使用场景详细对

Unity新手入门学习殿堂级知识详细讲解(图文)

《Unity新手入门学习殿堂级知识详细讲解(图文)》Unity是一款跨平台游戏引擎,支持2D/3D及VR/AR开发,核心功能模块包括图形、音频、物理等,通过可视化编辑器与脚本扩展实现开发,项目结构含A... 目录入门概述什么是 UnityUnity引擎基础认知编辑器核心操作Unity 编辑器项目模式分类工程

Go语言连接MySQL数据库执行基本的增删改查

《Go语言连接MySQL数据库执行基本的增删改查》在后端开发中,MySQL是最常用的关系型数据库之一,本文主要为大家详细介绍了如何使用Go连接MySQL数据库并执行基本的增删改查吧... 目录Go语言连接mysql数据库准备工作安装 MySQL 驱动代码实现运行结果注意事项Go语言执行基本的增删改查准备工作

Go中select多路复用的实现示例

《Go中select多路复用的实现示例》Go的select用于多通道通信,实现多路复用,支持随机选择、超时控制及非阻塞操作,建议合理使用以避免协程泄漏和死循环,感兴趣的可以了解一下... 目录一、什么是select基本语法:二、select 使用示例示例1:监听多个通道输入三、select的特性四、使用se