我使用 Pyramid 作为回合制视频游戏数据传输的基础。客户端使用 POST 数据来展示他们的动作,并使用 GET 来检索序列化的游戏板数据。游戏数据有时可能涉及字符串,但几乎总是两个整数和两个元组:
gamedata = (userid, gamenumber, (sourcex, sourcey), (destx, desty))
我的一般客户端框架是 Pickle ,转换为 base 64,使用 urlencode,然后提交 POST。然后服务器接收 POST,解压缩单项字典,解码 base64,然后解压缩数据对象。
我想使用 Pickle,因为我可以使用类和值。将游戏数据作为 POST 字段提交只能给我字符串。
然而,Pickle 被认为是不安全的。所以,我转向了 pyYAML,它具有相同的目的。使用yaml.safe_load(data)
,我可以在不暴露安全漏洞的情况下序列化数据。但是,safe_load 非常安全,我什至不能反序列化无害的元组或列表,即使它们只包含整数。
这里有中间立场吗?有没有一种方法可以序列化 python 结构而不同时允许执行任意代码?
我的第一个想法是为我的发送和接收函数编写一个包装器,在值名称中使用下划线来重新创建元组,例如发送会将字典值转换source : (x, y)
为source_0 : x, source_1: y
. 我的第二个想法是,这不是一个非常明智的发展方式。
编辑:这是我使用 JSON 的实现......它似乎不如 YAML 或 Pickle 强大,但我仍然担心可能存在安全漏洞。
在我进行实验时,客户端的构造更加明显:
import urllib, json, base64
arbitrarydata = { 'id':14, 'gn':25, 'sourcecoord':(10,12), 'destcoord':(8,14)}
jsondata = json.dumps(arbitrarydata)
b64data = base64.urlsafe_b64encode(jsondata)
transmitstring = urllib.urlencode( [ ('data', b64data) ] )
urllib.urlopen('http://127.0.0.1:9000/post', transmitstring).read()
Pyramid Server 可以检索数据对象:
json.loads(base64.urlsafe_b64decode(request.POST['data'].encode('ascii')))
在不相关的说明中,我很想听听关于在这种方法中使用 POST 数据的可接受性的其他意见,我的游戏客户端目前绝不是基于浏览器的。