1 # client.py
2 # ————————01CMDB获取服务器基本信息————————
3 from src import plugins #__init__.py
4 from lib.serialize import Json #转成字符串或者模式
5
6 # ————————02CMDB将服务器基本信息提交到API接口————————
7 import requests #伪造页面访问
8 from config import settings #文件配置
9 # ————————02CMDB将服务器基本信息提交到API接口————————
10
11 # ————————03CMDB信息安全API接口交互认证————————
12 import hashlib
13 import time
14 # ————————03CMDB信息安全API接口交互认证————————
15
16 # ————————04CMDB本地(Agent)模式客户端唯一标识(ID)————————
17 import os #操作系统层面执行
18 import shutil #高级的 文件、文件夹、压缩包 处理模块
19 # ————————04CMDB本地(Agent)模式客户端唯一标识(ID)————————
20 class AutoBase(object):
21
22 # ————————02CMDB将服务器基本信息提交到API接口————————
23 def __init__(self):
24 self.asset_api = settings.ASSET_API #ASSET_API = "http://127.0.0.1:8000/api/asset"
25
26 # ————————03CMDB信息安全API接口交互认证————————
27 self.key = settings.KEY # 用于API认证的KEY#KEY = '299095cc-1330-11e5-b06a-a45e60bec08b'
28 self.key_name = settings.AUTH_KEY_NAME # 'auth-key' API认证的请求头
29 # ————————03CMDB信息安全API接口交互认证————————
30
31 # ————————03CMDB信息安全API接口交互认证————————
32 def auth_key(self):#API接口认证
33 ha = hashlib.md5(self.key.encode('utf-8'))#认证的密码
34 time_span = time.time() #现在的时间戳 #1529819687.8867188
35 ha.update(bytes("%s|%f" % (self.key, time_span), encoding='utf-8'))#更新认证密码#密码+时间戳
36 encryption = ha.hexdigest() # 对‘时间密码’进行哈希
37 result = "%s|%f" % (encryption, time_span) #把‘时间密码’和 时间戳(解密用) 作为 API认证的请求头
38 print('‘时间密码’和 时间戳:',result)
39 return {self.key_name: result} # 'auth-key' API认证的请求头
40 # ————————03CMDB信息安全API接口交互认证————————
41
42 def post_asset(self, msg):#post方式向API接口提交资产信息
43 status = True#是否获取到信息
44 try:
45 # ————————03CMDB信息安全API接口交互认证————————
46 headers = {}
47 headers.update(self.auth_key())##认证的密码
48 # ————————03CMDB信息安全API接口交互认证————————
49 response = requests.post(
50 url=self.asset_api,
51 # ————————03CMDB信息安全API接口交互认证————————
52 headers=headers,
53 # ————————03CMDB信息安全API接口交互认证————————
54 json=msg
55 )
56 except Exception as e:
57 response = e
58 status = False #获取信息时出现错误
59 print(response.json())
60 # ————————02CMDB将服务器基本信息提交到API接口————————
61
62 def process(self):#派生类需要继承此方法,用于处理请求的入口
63 raise NotImplementedError('您必须实现过程的方法')
64
65 class AutoAgent(AutoBase):
66
67 # ————————04CMDB本地(Agent)模式客户端唯一标识(ID)————————
68 def __init__(self):
69 self.cert_file_path = settings.CERT_FILE_PATH# Agent模式保存服务器唯一ID的文件
70 super(AutoAgent, self).__init__()
71 """
72 super() 函数是用于调用父类(超类)的一个方法。
73 super 是用来解决多重继承问题的,直接用类名调用父类方法在使用单继承的时候没问题,
74 但是如果使用多继承,会涉及到查找顺序(MRO)、重复调用(钻石继承)等种种问题。
75 MRO 就是类的方法解析顺序表, 其实也就是继承父类方法时的顺序表。
76 """
77 def load_local_cert(self):# 获取本地唯一标识
78 if not os.path.exists(self.cert_file_path):#使用os.path.exists()方法可以直接判断文件(或者文件夹)是否存在
79 return None #如果文件(或者文件夹)不存在 #返回空
80 if os.path.isfile(self.cert_file_path): #判断是不是文件
81 with open(self.cert_file_path, mode='r') as f: #如果文件存在 #打开读取文件
82 data = f.read()#获取唯一ID的文件#read() 方法用于从文件读取指定的字节数,如果未给定或为负则读取所有。
83 if not data: # 如果文件没有内容 #返回空
84 return None
85 cert = data.strip() # strip() 方法用于移除字符串头尾指定的字符(默认为空格或换行符)或字符序列。
86 return cert # 返回 本地唯一标识
87 else: #防止有文件夹叫cert,导致无法创建cert文件
88 shutil.rmtree(self.cert_file_path)#就 递归的去删除文件夹 #这个目录以及子文件全部删除
89 return None #返回空 #去创建文件
90
91 def write_local_cert(self, cert):#写入本地唯一标识
92 with open(self.cert_file_path, mode='w') as f: #循环打开文件 #没有就创建 然后打开
93 f.write(cert) #写入本地唯一标识
94 # ————————04CMDB本地(Agent)模式客户端唯一标识(ID)————————
95
96 def process(self):
97 server_info = plugins.get_server_info()#获取本地基本信息
98
99 # ————————04CMDB本地(Agent)模式客户端唯一标识(ID)————————
100 if not server_info.status: #获取的基本信息是否出错
101 return
102 local_cert = self.load_local_cert() # 去获取本地唯一标识
103 if local_cert: #如果 获取本地唯一标识 后 #不是 return None
104 if local_cert == server_info.data['hostname']: #判断本地文件的唯一标识 是不是等于 获取的基本信息里主机名
105 pass #一样就什么都不干
106 else: #不一样就把 获取的基本信息里主机名 替换成 本地文件的唯一标识
107 #防止 恶意篡改 主机名 ,上报API时 错认为是 新资产
108 server_info.data['hostname'] = local_cert # 更新字典的主机名
109 else: # 获取本地唯一标识为None时 #把获取到的主机名写入本地做为唯一标识
110 self.write_local_cert(server_info.data['hostname'])
111 # ————————04CMDB本地(Agent)模式客户端唯一标识(ID)————————
112
113 server_json = Json.dumps(server_info.data)#json.dumps将 Python 对象编码成 JSON 字符串
114 print('提交资产信息:',server_json)
115 # ————————01CMDB获取服务器基本信息————————
116
117 # ————————02CMDB将服务器基本信息提交到API接口————————
118 self.post_asset(server_json)# post方式向接口提交资产信息
119 # ————————02CMDB将服务器基本信息提交到API接口————————