需求背景

蓝牙体征检测设备的初始上报频率非常高,单台每秒 370 个数据包。 需要通过通过服务器向蓝牙网关下发禁止波形数据的指令,禁用掉无用数据的上报。

有两个禁用策略:

  • 定时下发禁用指令。例如每十秒
  • 收到波形数据时,就立即下发禁用指令

显然第二种方式更合理,而且在有多台设备接入的情况下,也方便批量下发(根据 mac 地址)。

但是,这里就出现了一个频率控制问题,就是从下发指令,到禁止成功,是有一个时间间隔的。这个时间间隔内,要规避重复下发指令。

同时,还有另外一个需求,就是对写入数据进行控制,设备方的体温上报频率过高,增加了存储成本,所以同样需要限制。

找到一个 golang 的官方库,可以方便的实现这个频率控制功能:

go get golang.org/x/time/rate

类似场景:IP 限速

https://medium.com/@pliutau/rate-limiting-http-requests-in-go-based-on-ip-address-4e66d1bea4cf

原理就是:

使用 rate 库,每个 ip 对应一个 limiter。对应的这里使用 mac 地址作为 key。

内存控制

如果不断的加 IP,如何控制总内存。

限制 IP 总量,当超过时,清掉长期不用的。虽然目前的应用场景,设备总量可控,而且不太可能超过 1000。

但,这个很容易变成一个漏洞。如果黑客模拟发包,导致 key 总量不可控,内存爆炸。。。是否可以像 nginx 一样,限制内存使用上限。

限速一秒一次

除了直接指定每秒产生的 Token 个数外,还可以用 Every 方法来指定向 Token 桶中放置 Token 的间隔,例如:

limit := rate.Every(1 * time.Second);
limiter := rate.NewLimiter(limit, 1);

https://www.cyhone.com/articles/usage-of-golang-rate/

如何测试

  • 一个高频的定时调用 (调用者)
  • 一个低频限速的函数(被调用者)

加锁是否会影响性能

频繁 RWMutex Lock 是否影响性能。

不必担心。RWMutex 是读写锁,同时,Lock 是加写锁,RLock 是加读锁,性能更加有保证了。

互斥锁 (sync.Mutex) 和读写锁 (sync.RWMutex)

https://geektutu.com/post/hpg-mutex.html

  • 互斥锁:互斥即不可同时运行。即, 使用了互斥锁的两个代码片段互相排斥,只有其中一个代码片段执行完成后,另一个才能执行。
  • 读写锁:保证读操作的安全,那只要保证并发读时没有写操作在进行就行。在这种场景下我们需要一种特殊类型的锁,其允许多个只读操作并行执行,但写操作会完全互斥。

TODO

  • [X] 收到波形数据包时,调用禁用函数,向网关下发禁用指令
  • [X] 禁用函数,使用统一 client id
  • [X] 禁用函数,接受一个结构体参数,里面包含体征蓝牙设备 mac 地址
  • [X] 结构体增加字段:体征设备 mac 地址
  • [X] 网关监听 mqtt topic 的名称规范,用于接受下发指令: 监听 topic: healthdata/gateway/2207027001
  • [X] 提取 mac 地址,并拼接下发指令。指令测试格式在单元测试文件里。
  • [X] 调整三种波形禁用指令的发送间隔, 100ms
  • [X] 在禁用函数内,增加频率控制的逻辑。限制一秒一次
  • [X] 修改测试设备,网关 mqtt topic
  • 数据写入时的频率控制 (优先级低,可以回去台式机上测试,再部署),每个指标一个 rate limiter 感觉太啰嗦了。不知道 influxdb 是否支持