Python3调用Salt API
curl调用salt api cmd.run使用方法如下:
curl http://10.10.2.11:8000 \ -H 'Accept: application/x-yaml' \ -H 'X-Auth-Token: 042a226e6'\ -d client=local \ -d tgt='10.10.2.4' \ -d fun='cmd.run' \ -d arg='get-host' curl http://10.10.2.11:8000 \ -H 'Content-Type: application/json' \ -H 'X-Auth-Token: 042a226e6'\ -d '[{"client": "local", "tgt": "10.10.2.4", "fun": "cmd.run","arg":"get-host" ,"kwarg":{"shell": "powershell"} }]' curl http://10.10.2.11:8000 \ -H 'Accept: application/x-yaml' \ -H 'X-Auth-Token: 042a226'\ -d client=local \ -d tgt='10.10.2.4' \ -d fun='cmd.run' \ -d arg='get-host' \ -d arg='shell="powershell"'
附:
There is already an option to use JSON in requests to salt-api. This is the best (easiest) way to send complex data structures in the HTTP request. It looks like this:
curl -k https://localhost:8000/run \
[...snip...] \
-H 'Content-Type: application/json' \
-d '[{"client": "local", "tgt": "*", "fun": "test.arg", "kwarg": {"a": 1}}]'
If you would prefer to continue using the urlencoded format instead of JSON there is an odd trick you can use since salt-api and the salt
CLI program both use Salt's Python API and any CLI parameters are run through the YAML parser. The syntax is a bit rough but it looks like this:
curl -k https://localhost:8000/run \
[...snip...] \
-d arg='a=1'
Note that this isn't an intended feature. It's a side effect of how Salt's Python API and the CLI work together. You're much better off sending real JSON if you can.
Python3调用salt api方法如下:
import urllib.request,urllib.parse,json class saltAPI(): def __init__(self): self.url = 'http://10.10.2.11:8000' self.data = {'username' : 'salt-api', 'password' : 'salt-api', 'eauth' : 'pam'} self.headers = {'User-Agent':'Mozilla/5.0 (X11; Fedora; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/58.0.3029.110 Safari/537.36'} self.token = self.get_token() self.headers['X-Auth-Token'] = self.token #获取token def get_token(self): postdata = urllib.parse.urlencode(self.data).encode('utf8') content = self.post_request('/login',postdata) try: token = content['return'][0]['token'] return token except Exception as ErrorMsg: print(ErrorMsg) #发送post def post_request(self,prefix='/',postdata=None): try: request = urllib.request.Request(self.url+ prefix,postdata,headers=self.headers) reponse = urllib.request.urlopen(request).read() content = json.loads(reponse.decode('utf8')) return content except Exception as ErrorMsg: print(ErrorMsg) exit() #执行远程命令 def SaltCmd(self,tgt,fun='cmd.run',arg=None): #通过urlencode格式提交参数,则会多个arg,dict中不能包含重复key,需要将其转码为bytes后重新拼接 params = {'client':'local', 'tgt':tgt, 'fun':fun, 'arg':arg} params2 = {'arg':'shell=powershell'} obj = urllib.parse.urlencode(params).encode('UTF8') obj2 = urllib.parse.urlencode(params2).encode('UTF8') obj_all = obj + b'&' +obj2 ret = self.post_request(prefix='/',postdata=obj_all) res = ret['return'][0] return(res[tgt].replace('\r','')) #执行远程命令1,功能同上 def SaltCmdrun(self,tgt,fun='cmd.run',arg=None): #通过json格式提交参数,需要指定Content-Type(服务器/客户端提交的数据类型),Accept表示客户端希望接收的类型;这里指定Shell=PowerShell self.headers['Content-Type'] = 'application/json' params = [{'client': 'local', 'tgt': tgt, 'fun': fun,'arg':arg ,'kwarg':{'shell': 'powershell'} }] #json.loads() transform str to dict;json.dumps() transform dict to str. #将params转换为浏览器可识别的bytes类型 obj = bytes(json.dumps(params),'utf8') ret = self.post_request(prefix='/',postdata=obj) res = ret['return'][0] return(res[tgt].replace('\r','')) #执行远程命令2,功能同上 def SaltCmdrun2(self,tgt,fun='cmd.run',arg=None,**kwargs): #通过json格式提交参数,需要指定Content-Type self.headers['Content-Type'] = 'application/json' params = [{'client': 'local', 'tgt': tgt, 'fun': fun,'arg':arg ,'kwarg':kwargs }] #json.loads() transform str to dict;json.dumps() transform dict to str. #将params转换为浏览器可识别的bytes类型 obj = bytes(json.dumps(params),'utf8') ret = self.post_request(prefix='/',postdata=obj) res = ret['return'][0] return(res[tgt].replace('\r','')) #打印Header(含token) def test(self): print(self.headers) s = saltAPI() # s.test()
t1 = s.SaltCmd('10.10.2.4','cmd.run','get-host') print(t1) # t1 = s.SaltCmdrun('10.10.2.4','cmd.run','get-host') # print(t1) # t2 = s.SaltCmdrun2('10.10.2.4','cmd.run','gwmi win32_bios',shell='powershell') # print(t2)
#多台主机同时执行cmd.script,使用expr_form=list,脚本参数通过args传递 def SaltCmd_Multi(self,tgt,fun='cmd.run',expr_form='glob',arg=None,**kwargs): #通过json格式提交参数,需要指定Content-Type self.headers['Content-Type'] = 'application/json' params = [{'client': 'local', 'tgt': tgt, 'fun': fun,'expr_form':expr_form,'arg':arg ,'kwarg':kwargs }] print(params) #json.loads() transform str to dict;json.dumps() transform dict to str. #将params转换为浏览器可识别的bytes类型 obj = bytes(json.dumps(params),'utf8') ret = self.post_request(prefix='/',postdata=obj) print(ret) t2 = s.SaltCmd_Multi('10.10.2.4,10.10.2.5','cmd.script','list','http://10.10.2.6/OService.ps1',args='w32time Query',shell='powershell')
t2 = s.SaltCmd_Multi('10.10.2.4','cmd.script','glob','http://10.10.2.6/OService.ps1',args='w32time Query',shell='powershell')
print(t2)