tt-server/server.py

224 lines
5.4 KiB
Python
Raw Normal View History

2023-03-11 08:04:43 +00:00
#!/usr/bin/env python3
2023-03-10 01:34:44 +00:00
#coding:utf-8
import socketserver
import subprocess
2023-03-11 03:51:56 +00:00
import pickle
2023-03-10 09:58:19 +00:00
offs = 0 #偏移量
2023-03-10 01:34:44 +00:00
p ={}
#保存终端信息ccid与地址对
#p={ccid=addr,}
#缓存未能正常发送消息的客户端
2023-03-10 09:58:19 +00:00
pp={}
2023-03-11 03:51:56 +00:00
# cache={}
from collections import defaultdict
cache=defaultdict(list)
2023-03-10 01:34:44 +00:00
2023-03-11 03:51:56 +00:00
def save_cache(dic):
2023-03-11 08:04:43 +00:00
'''缓存未发送成功的消息
仅在未发送成功时缓存'''
2023-03-11 03:51:56 +00:00
# f=open("./cache.txt",mode='w+')
# f.write(str(dic))
# f.close
pickle.dump(dic,open("./cache.txt",'wb'))
def load_cache():
2023-03-11 08:04:43 +00:00
'''程序启动时加载缓存'''
2023-03-11 03:51:56 +00:00
try:
# f=open("./cache.txt")
# c = eval(f.read())
# f.close()
c=pickle.load(open("./cache.txt",'rb'))
return c
except:
return None
2023-03-10 09:58:19 +00:00
def update_pairs(addr,ccid,clear=0):
'''更新终端库
addr:绑定tcp地址
ccid:终端号
clear:清空掉线终端'''
if ccid and not clear:
#添加或更新客户端
p[ccid] = addr
pp[addr] = ccid
else:
del p[ccid]
del pp[addr]
# 打印在线终端
for i in p:
print(i,"<--->",p[i])
# for i in pp:
# print(i,"-",pp[i])
def get_addr(ccid):
'''获取ccid对应的地址'''
2023-03-10 01:34:44 +00:00
try:
2023-03-10 09:58:19 +00:00
# 如果有直接返回
return p[ccid]
2023-03-10 01:34:44 +00:00
except Exception:
2023-03-10 09:58:19 +00:00
#没有就返回None
return None
def get_ccid(addr):
try:
# 如果有直接返回
return pp[addr]
except Exception:
#没有就返回None
return None
'''
+--------+--------------+----------+----------+--------------+------------------------+--------+
| ff | 长度 | 地址信息 | 信令 | 内容 | 校验位 |
+--------+--------------+----------+----------+--------------+------------------------+--------+
| | 2Bytes | 4Bytes | 心跳 | 0x01 | 时间戳32bit单位ms | 2Bytes |
+ $TTMS +--------------+----------+----------+--------------+------------------------+--------+
| | 2Bytes | 4Bytes | 传输数据 | 0xaa | 载荷 | 2Bytes |
+--------+--------------+----------+----------+--------------+------------------------+--------+
'''
2023-03-11 03:51:56 +00:00
def send(fd,data):
try:
fd.send(data)
return 0
except:
return -1
2023-03-10 09:58:19 +00:00
def tt_hh(addr,data):
'''处理心跳'''
# 更新pairs
2023-03-11 03:51:56 +00:00
ccid = data[7+offs:11+offs]
update_pairs(addr,ccid)
2023-03-10 09:58:19 +00:00
# 组帧再返回
2023-03-11 03:51:56 +00:00
# 从缓存中匹配数据并重发
2023-03-11 08:04:43 +00:00
# 仅有新终端上线时才重发
2023-03-11 03:51:56 +00:00
if ccid in cache:
2023-03-11 08:04:43 +00:00
msgs = cache[ccid]
for i in range(len(msgs)):
try:
addr.send(msgs[i])
# 发送成功则清除对应缓存数据
del cache[ccid][i] # not msgs[i]
except:
# 异常时退出循环
break
2023-03-10 09:58:19 +00:00
return 0
def tt_trans(addr,data):
'''发送数据到指定ccid'''
tccid = data[7+offs:11+offs]
taddr = get_addr(tccid)
# 组帧
msg = bytearray(data[:])
2023-03-11 03:51:56 +00:00
msg[:5]=b'$TTMX'
2023-03-10 09:58:19 +00:00
msg[7+offs:11+offs] = get_ccid(addr)
# 计算校验和
crc = 0
2023-03-11 03:51:56 +00:00
for i in msg[:-1]:
2023-03-10 09:58:19 +00:00
crc = crc ^ i
msg[-1]=crc
2023-03-11 03:51:56 +00:00
if not taddr:
2023-03-11 08:04:43 +00:00
# 未报到匹配的在线终端
2023-03-11 03:51:56 +00:00
#更新缓存
#{tccid1=[msg1,msg2,...],...}
cache[tccid].append(msg)
save_cache(cache)
2023-03-11 08:04:43 +00:00
print("终端未在线或ccid错。")
2023-03-11 03:51:56 +00:00
return 0
2023-03-11 08:04:43 +00:00
print("匹配终端:",tccid,"<--->",taddr)
2023-03-11 03:51:56 +00:00
2023-03-10 09:58:19 +00:00
# 发送
try:
taddr.send(msg)
except Exception:
2023-03-11 08:04:43 +00:00
# 发送失败
2023-03-11 03:51:56 +00:00
# 更新pairs清空对应终端
2023-03-10 09:58:19 +00:00
update_pairs(taddr,tccid,1)
2023-03-11 03:51:56 +00:00
#更新缓存
#{tccid1=[msg1,msg2,...],...}
cache[tccid].append(msg)
save_cache(cache)
2023-03-11 08:04:43 +00:00
print("发送失败,终端可能已掉线。",taddr)
2023-03-10 09:58:19 +00:00
return 0
def err_handle(flag,addr):
'''错误处理
默认打印错误信息
1-非法连接关闭连接
'''
return 0
def tt_decode(addr,data):
'''处理收到的数据'''
2023-03-11 03:51:56 +00:00
if data[:5] != b'$TTMS' or len(data) <12:
2023-03-10 09:58:19 +00:00
# 数据长度不可能小于12考虑非法接入的可能主动断开连接
return -1
cmd = data[11+offs]
if cmd==0x01:
# 处理心跳
tt_hh(addr,data)
elif cmd==0xaa:
# 处理数据
tt_trans(addr,data)
else:
return -1
2023-03-10 01:34:44 +00:00
class MyServer(socketserver.BaseRequestHandler):
def handle(self):#回调
print ("终端已上线:",self.client_address)
conn = self.request
2023-03-10 09:58:19 +00:00
# print(type(conn),conn.fd)
2023-03-10 01:34:44 +00:00
while True:
data = conn.recv(200)
2023-03-10 09:58:19 +00:00
# update_pairs(conn,data[7+offs:11+offs])
2023-03-10 01:34:44 +00:00
if not data:
break
print ("接收到新数据",self.client_address,",长度",len(data),"\r\n", data.hex(" "))
#ack_msg = "got from "+ str(ip) + " to " + str(self.client_address) + data
2023-03-10 09:58:19 +00:00
# conn.send(data[:6])
if tt_decode(conn,data) :
conn.close()
print(self.client_address,"疑是非法连接,已切断。")
break
2023-03-10 01:34:44 +00:00
if __name__ == '__main__':
2023-03-11 03:51:56 +00:00
# load_cache
cache = load_cache()
2023-03-10 01:34:44 +00:00
server = socketserver.ThreadingTCPServer(('localhost',8005), MyServer)
ip, port = server.server_address
print ("服务端已建立:",ip, port)
server.serve_forever()
2023-03-11 03:51:56 +00:00