1
0
mirror of https://gitee.com/willfree/mlsr.git synced 2026-06-15 20:04:48 +08:00
Files
mlsr/hello/HelloProtocol.go
T

287 lines
10 KiB
Go

// Package hello
// @Author: Pei Xinyuan
// @Description:
// @Version: 0.1.0
// @Date: 2022/7/14 14:49
// @Copyright: MIN-Group;国家重大科技基础设施——未来网络北大实验室;深圳市信息论与未来网络重点实验室
//
package hello
import (
common2 "minlib/common"
"minlib/component"
"minlib/logicface"
"minlib/packet"
"minlib/security"
"mlsr/common"
"mlsr/lsa"
"mlsr/lsdb"
"mlsr/route"
"mlsr/utils"
)
// 用于hello兴趣包的名称的组件
const INFO_COMPONENT = "INFO"
const MLSR_COMPONENT = "mlsr"
// 正常收到数据包的触发函数
type DataCallback = func(interest *packet.Interest, data *packet.Data)
// 收到NACK的触发函数
type NackCallback = func(interest *packet.Interest, nack *packet.Nack)
// 超时未响应的触发函数
type TimeoutCallback = func(interest *packet.Interest)
//type OptionalNack = *packet.Nack
type HelloProtocol struct {
m_face *logicface.LogicFace
m_scheduler *lsdb.MlsrScheduler
m_keychain *security.KeyChain
mlsrConfig *common.MlsrConfig
m_routingTable *route.RoutingTable
m_lsdb *lsdb.Lsdb
m_adjacencyList *lsa.AdjLsaAdjacenctList
// 收包池
receivePacketQue chan *packet.Interest // 从其它路由器收包后,放入该队列
state bool // 程序生成状态, true 为正常运行, false 为关闭状态
// 接收到hello响应数据包后的信号量触发
OnInitialHelloDataValidated *utils.Signal
}
//
// Init
// @Description: Hello协议-初始化
// @receiver hp
// @param face
// @param scheduler
// @param keychain
// @param mlsrConfig
// @param routingTable
// @param mlsdb
//
func (hp *HelloProtocol) Init(face *logicface.LogicFace,
scheduler *lsdb.MlsrScheduler,
keychain *security.KeyChain, mlsrConfig *common.MlsrConfig,
routingTable *route.RoutingTable, mlsdb *lsdb.Lsdb) {
// 信号量初始化
hp.OnInitialHelloDataValidated = utils.NewSignal()
// 其它数据结构
hp.m_face = face
hp.m_scheduler = scheduler
hp.m_keychain = keychain
hp.mlsrConfig = mlsrConfig
hp.m_routingTable = routingTable
hp.m_lsdb = mlsdb
hp.m_adjacencyList = mlsrConfig.GetAdjacencys()
// logicface相关
// 构造接收hello兴趣包的前缀
name, err := component.CreateIdentifierByString(
hp.mlsrConfig.GetRouterPrefix().ToUri() + "/" + MLSR_COMPONENT + "/" + INFO_COMPONENT)
if err != nil {
common2.LogFatal("HelloProtocol Setting interest filter for Hello interest error, interest generate wrong.", err)
return
}
// 设置监听前缀
err = hp.m_face.RegisterPullIdentifier(name, 3000)
if err != nil {
common2.LogFatal("HelloProtocol Setting interest filter for Hello interest error, identifier register wrong.", err)
return
}
// 开一个线程去接收前缀为hello协议所监听前缀的兴趣包,并进行处理
hp.receivePacketQue = make(chan *packet.Interest, 1000)
hp.state = true
// 启动处理接收到的包的协程
go hp.dealWithRecvPktRoutine()
// 启动从客户端接收包的协程
go hp.recvPacketFromClientRoutine()
}
//
// expressInterest
// @Description: 根据传入的兴趣包名称和兴趣包过期时间,构建兴趣包,并发出去
// @receiver hp
// @param interestName
// @param _millseconds
//
func (hp *HelloProtocol) expressInterest(interestName *component.Identifier, _millseconds uint32) {
common2.LogDebug("HelloProtocol Expressing Interest: " + interestName.ToUri())
interest := packet.NewInterest()
interest.SetName(interestName)
interest.SetInterestLifeTime(uint64(_millseconds))
// todo 这里需要实现的是,将兴趣包通过face发出,并对超时、nack、正常应答三种情况进行处理
}
//
// SendHelloInterest
// @Description: 向指定邻居路由器发送hello探测包
// @receiver hp
// @param neighbor
//
func (hp *HelloProtocol) SendHelloInterest(neighbor *component.Identifier) {
// 查找关于某个邻居路由器的邻接信息
adjInfo := hp.m_adjacencyList.FindAdjacencyByRouterIdentifier(neighbor)
// 如果没有找到,则不做处理
if adjInfo == nil {
return
}
// 如果找到了,且有对应的faceID,则发送一个hello探测包
if adjInfo.LogicFaceId() != 0 {
// interest name: /<neighbor>/NLSR/INFO/<router>
name, err := component.CreateIdentifierByString(adjInfo.GetNeighborRouterIdentifier().ToUri() +
"/" + MLSR_COMPONENT + "/" + INFO_COMPONENT +
"/" + GetSinglePrefixForHelloProtocol(hp.mlsrConfig.GetRouterPrefix()))
if err != nil {
common2.LogDebug("HelloProtocol SendHelloInterest error, hello name create wrong.", err)
return
}
// 将data发出去
hp.expressInterest(name, hp.mlsrConfig.GetInterestResendTime())
common2.LogDebug("HelloProtocol Sending HELLO interest: " + name.ToUri())
}
// 安排按照配置文件中的时间,若干秒后【默认一分钟】发送下一个hello兴趣包
job, err := hp.m_scheduler.ScheduleTaskAfterDuration(hp.mlsrConfig.GetInfoInterestInterval(), 1).
Do(hp.SendHelloInterest, neighbor)
job.LimitRunsTo(1)
hp.m_scheduler.StartAsync()
if err != nil {
common2.LogError("HelloProtocol ScheduleTask to send hello interest error. ", err.Error())
}
}
//
// processInterestTimedOut
// @Description: 发送兴趣包给邻居路由器,当超时或nack时,执行本函数
// @receiver hp
// @param interest
//
func (hp *HelloProtocol) processInterestTimedOut(interest *packet.Interest) {
// interest name: /<neighbor>/NLSR/INFO/<router>
interestName := interest.GetName()
// 取出兴趣包倒数第二个组件,对其进行校验
infoComponent, err := interestName.Get(-2)
if err != nil {
common2.LogDebug("HelloProtocol processInterestTimedOut error, get(-2) err ", err)
return
}
if infoComponent.ToUri() != INFO_COMPONENT {
common2.LogDebug("HelloProtocol processInterestTimedOut error, INFO_COMPONENT not found in interestName ", err)
return
}
// 取"/NLSR"之前的所有组件组成的前缀,也即邻居路由器的前缀
neighborIdentifier, err := interestName.GetPrefix(-3)
if err != nil {
common2.LogDebug("HelloProtocol processInterestTimedOut error, interestName.GetPrefix error.", err)
return
}
// 将本机邻居列表中该邻居的超时次数增加1
err = hp.m_adjacencyList.IncrementTimedOutInterestCount(neighborIdentifier)
if err != nil {
common2.LogDebug("HelloProtocol processInterestTimedOut error, IncrementTimedOutInterestCount error.", err)
return
}
// 查找该邻居的状态
status, err := hp.m_adjacencyList.GetStatusOfNeighbor(neighborIdentifier)
if err != nil {
common2.LogDebug("HelloProtocol processInterestTimedOut error, GetStatusOfNeighbor error.", err)
return
}
// 查找该邻居的超时次数
infoIntTimedOutCount, err := hp.m_adjacencyList.GetTimedOutInterestCount(neighborIdentifier)
if err != nil {
common2.LogDebug("HelloProtocol processInterestTimedOut error, GetTimedOutInterestCount error.", err)
return
}
common2.LogDebug("HelloProtocol processInterestTimedOut Status: ", status)
common2.LogDebug("HelloProtocol processInterestTimedOut TimeoutCount: ", infoIntTimedOutCount)
// 如果超时次数小于阈值。则继续探测
if infoIntTimedOutCount < hp.mlsrConfig.GetInterestResendTime() {
name, err := component.CreateIdentifierByString(neighborIdentifier.ToUri() +
"/" + MLSR_COMPONENT + "/" + INFO_COMPONENT +
"/" + GetSinglePrefixForHelloProtocol(hp.mlsrConfig.GetRouterPrefix()))
if err != nil {
common2.LogDebug("HelloProtocol processInterestTimedOut error, hello name create wrong.")
return
}
common2.LogDebug("HelloProtocol processInterestTimedOut Resending interest: " + interestName.ToUri())
// 将兴趣包再次发出去
hp.expressInterest(name, hp.mlsrConfig.GetInterestResendTime())
} else if status == lsa.STATUS_ACTIVE {
// 如果重传次数达到阈值,且该路由器旧状态为激活状态,则将其修改为未激活
err = hp.m_adjacencyList.SetStatusOfNeighbor(neighborIdentifier, lsa.STATUS_INACTIVE)
if err != nil {
common2.LogDebug("HelloProtocol processInterestTimedOut error, set status error. ", err)
}
common2.LogDebug("HelloProtocol processInterestTimedOut, change status of " +
neighborIdentifier.ToUri() + " to IN-ACTIVE")
// 安排lsdb的邻接lsa重新构建
hp.m_lsdb.ScheduleAdjLsaBuild()
}
}
//
// onContent
// @Description: 发送兴趣包给邻居路由器,当收到data包时,执行本函数
// @receiver hp
// @param data
//
func (hp *HelloProtocol) onContent(interest *packet.Interest, data *packet.Data) {
common2.LogDebug("HelloProtocol Received data for INFO(name): " + data.GetName().ToUri())
// todo 这里先跳过签名验证,执行验证通过后的处理
// 取出数据包名称的倒数第三个组件,对其进行校验
infoComponent, err := data.GetName().Get(-3)
if err != nil {
common2.LogDebug("HelloProtocol onContent error, get(-3) err ", err)
return
}
if infoComponent.ToUri() != INFO_COMPONENT {
common2.LogDebug("HelloProtocol onContent error, INFO_COMPONENT not found in dataName ", err)
return
}
// 取"/NLSR"之前的所有组件组成的前缀,也即邻居路由器的前缀
neighborIdentifier, err := data.GetName().GetPrefix(-4)
if err != nil {
common2.LogDebug("HelloProtocol onContent error, dataName.GetPrefix error.", err)
return
}
// 查找该邻居的状态
oldStatus, err := hp.m_adjacencyList.GetStatusOfNeighbor(neighborIdentifier)
if err != nil {
common2.LogDebug("HelloProtocol onContent error, get old status error.", err)
return
}
// 设置该路由器的状态及超时次数
err = hp.m_adjacencyList.SetStatusOfNeighbor(neighborIdentifier, lsa.STATUS_ACTIVE)
if err != nil {
common2.LogDebug("HelloProtocol onContent error, set status error.", err)
return
}
err = hp.m_adjacencyList.SetTimedOutInterestCount(neighborIdentifier, 0)
if err != nil {
common2.LogDebug("HelloProtocol onContent error, set timeout count error.", err)
return
}
// 查找该邻居的新状态
newStatus, err := hp.m_adjacencyList.GetStatusOfNeighbor(neighborIdentifier)
if err != nil {
common2.LogDebug("HelloProtocol onContent error, get new status error.", err)
return
}
common2.LogDebug("HelloProtocol onContent, neighbor: ", neighborIdentifier.ToUri(),
", old status: ", oldStatus, ", new status: ", newStatus)
// 如果状态发生了更改,则安排邻接Lsa进行重新构建,并发射信号给fib,使其更新mir中的该face状态
if newStatus != oldStatus {
hp.m_lsdb.ScheduleAdjLsaBuild()
hp.OnInitialHelloDataValidated.Emit(neighborIdentifier)
}
}