import base64 import hashlib import json import os.path import re import requests from Crypto.Cipher import AES from config import TARGET_URL from tools.logger_handle import logger class ServerClient: def __init__(self, server_host, user_name, password): self.user_name = user_name self.password = password self.serve_host = server_host self.access_token = None self.login_reply = None self.headers = { 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/133.0.0.0 Safari/537.36', } self.login() def login(self): url = self.serve_host + '/auth/oauth2/token' reply = requests.post(url, params={ 'username': self.user_name, 'password': self.encryption(self.password), 'grant_type': 'password', 'scope': 'server', 'client_id': 'knowledge', 'client_secret': 'knowledge' }, headers={'Content-Type': 'multipart/form-data', 'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/133.0.0.0 Safari/537.36'}) self.access_token = reply.json()['access_token'] self.login_reply = reply.json() self.headers['Authorization'] = f'Bearer {self.access_token}' def decryption(self, hex_str, secret='anZz000000000000'): key_bytes = secret.encode('utf-8') iv = key_bytes base64_str = base64.b64encode(bytes.fromhex(hex_str)).decode('utf-8') cipher = AES.new(key_bytes, AES.MODE_CBC, iv) decrypted = cipher.decrypt(base64.b64decode(base64_str)) try: decoded = decrypted.decode('utf-8') except UnicodeDecodeError: decoded = decrypted.decode('utf-8', errors='ignore') # 清除控制字符(类似 JS 中的 [\u0000-\u001F\u007F-\u009F]) cleaned = re.sub(r'[\x00-\x1F\x7F-\x9F]', ' ', decoded) try: return json.loads(cleaned) except json.JSONDecodeError: return cleaned.strip() def zero_pad(self, data, block_size=16): pad_len = block_size - (len(data) % block_size) if pad_len == 0: return data return data + b'\x00' * pad_len def encryption(self, payload, secret='a25vd2xlZGdl0000'.encode('utf-8')): data_bytes = payload.encode('utf-8') padded_data = self.zero_pad(data_bytes) cipher = AES.new(secret, AES.MODE_CBC, secret) encrypted = cipher.encrypt(padded_data) return encrypted.hex() def get_base_path(self): url = self.serve_host + '/mgr/document/dcLibrary/knowledges/owner?size=1000¤t=1' reply = requests.get(url, headers=self.headers) reply_data = reply.json() if reply_data.get('code') != 0: return [] data = self.decryption(reply_data['data']) return data.get('records', []) def get_folder_by_id(self, _id): url = f'{self.serve_host}/mgr/document/dcLibrary/tree?id={_id}' self.headers['documentid'] = _id reply = requests.get(url, headers=self.headers) reply_data = reply.json() if reply_data.get('code') != 0: return [] data = self.decryption(reply_data['data']) return data.get('data', []) def check_resp(self, resp): if resp.json()['code'] == -2: self.login() return True return False def download_file(self, file_info, storage_path): try: resp = requests.get(f'{self.serve_host}/mgr/document/dcLibrary/file/get/file/{file_info["id"]}', headers=self.headers) local_file_name = file_info['filePath'].replace('/', '_') with open(os.path.join(storage_path, local_file_name), 'wb') as f: f.write(resp.content) with open(os.path.join(storage_path, local_file_name + '.metadata'), 'w', encoding='utf-8') as f: f.write(json.dumps({ 'file_id': file_info["id"], 'bucket_name': file_info['bucketName'], 'md5': self.get_file_md5(os.path.join(storage_path, local_file_name)), 'file_name': file_info['name'], 'oss_path': file_info['filePath'], 'update_time': file_info['updateTime'] })) except Exception as e: logger.exception(e) return False return True def upload_file(self, file_path): try: with open(f'{file_path}.metadata', 'r', encoding='utf-8') as f: file_metadata = json.loads(f.read()) file_id = file_metadata['file_id'] files = [('file', (file_metadata['file_name'], open(file_path, 'rb'), 'application/vnd.openxmlformats-officedocument.wordprocessingml.document'))] resp = requests.post( f'{self.serve_host}/mgr/document/dcLibrary/saveByPython/{file_id}', headers=self.headers, files=files, data={} ) if self.check_resp(resp): resp = requests.post( f'{self.serve_host}/mgr/document/dcLibrary/saveByPython/{file_id}', headers=self.headers, files=files, data={}) if resp.json()['code'] == 0: return True else: logger.error(resp.json()) return False except Exception as e: logger.exception(e) return False @staticmethod def get_file_md5(file_path): md5 = hashlib.md5() with open(file_path, 'rb') as f: # 分块读取以支持大文件 while chunk := f.read(8192): md5.update(chunk) return md5.hexdigest() @staticmethod def load_metadata(local_path): with open(f'{local_path}.metadata', 'r', encoding='utf-8') as f: file_metadata = json.loads(f.read()) return file_metadata def create_cloud_file(self, local_file='D:/bussion/法律法规/道路交通安全法.docx', folder_id='5b091f1007c880528bd913530987d5e5'): payload = {'module': '/jvs-knowledge-ui/jvs-knowledge-import/'} files = [ ('file', (os.path.basename(local_file), open(local_file, 'rb'), 'application/vnd.openxmlformats-officedocument.wordprocessingml.document'))] resp = requests.post(f'{TARGET_URL}/mgr/document/dcLibrary/upload/{folder_id}', headers=self.headers, data=payload, files=files) if resp.json().get('code') == 0: return True else: return False def get_folder_tree(self): resp = requests.get(f'{TARGET_URL}/mgr/document/dcLibrary/tree/myFolder', headers=self.headers) if resp.json().get('code') == 0: return self.decryption(resp.json()['data']) else: return [] def get_template_file_info(self, file_id): try: resp = requests.get(f'{TARGET_URL}/mgr/document/dcLibrary/template/get/{file_id}', headers=self.headers) if resp.json().get('code') == 0: return self.decryption(resp.json()['data']) return False except Exception as e: logger.error('获取模板信息出错') logger.exception(e) return False def download_template(self, url, file_path): try: resp = requests.get(url) with open(file_path, 'wb') as f: f.write(resp.content) return True except Exception as e: logger.error('下载模板文件出错') logger.exception(e) return False def get_file_info(self, file_id): ''' :param file_id: :return: ''' logger.info(self.headers) try: resp = requests.get(f'{self.serve_host}/mgr/document/no/auth/dcLibrary/info/{file_id}', headers=self.headers) if not self.check_resp(resp): resp = requests.get(f'{self.serve_host}/mgr/document/no/auth/dcLibrary/info/{file_id}', headers=self.headers) if not resp.json()['data']: return False return self.decryption(resp.json()['data']) except Exception as e: logger.exception(e) return False if __name__ == '__main__': client = ServerClient(f'{TARGET_URL}', 'admin', 'jxkj123456') res = client.decryption( "9f79aa09fbeba3e4767bab05c6081c3d95d5dfd297ea65c58484ed5bc37b91cc86f731abb9e7c50555f6029f6d210b433a8821dbeaf4a7768688d4054076f7342f2a8a6db59163d0c6c14523f84a850f1f9c06334a92ee5a56e1a1752930773178fbea2b1219c076aced0bd2e4114f3564acf9ee5f3fa91cff8582010be8aa5a11e66acaeb24b890af691cfba58f86d86680f64753137637cd4f98432f8bb48ec653dc3474d30d20443a9f42a677d2cc6b777530df6d41302a60e4a033aa55731df7149774860fdb5c7855e3f94e657672be15c5e03cdd5e1288bc43d3b10f87d358a8d82807bc26955ad9173a30f81d8d11615906ceb4b007bf8009ff4c50c026b450ac28a3a3030a918197fc3f7e6b350a94a634f5954d945f25d419adf3d67caf9ac144f86a0984e608fa84a65e127845edeece7d9b3e381bc3a01db06d075514da65c721c0eca4d88a344850d5f7f53bdf5f83c29a9dfb189e300e387bfab1dedceea2ad1c7c17e4a5628f385059978e2870eb0c379f08956755043c62bc9e648383fc5e341d8a6f1c75be38b17e593ec264397dbca67855868f61957b8253c189ae04e470982e583bce7083db3357f1206490ea69ac023fce58a497ef37739d9e018345f481495d21c9dd91ae247ad72eb15efab7cab9177806a1c77c2e66f7c69e1617b6272dfcabc756078f5ee8237e112b8da2de64bce0d09f7eea540e5006e0634211998161bf8d1da451dfaa8e3b41d7b001d28821f906106c81eed3c50da52ee6169bd0df26182944fbc7e549badd25171b08202bf02f5dae7f2673ef156e61f5167efe2f10646feaccc295d0d5699e40b5c5543dd1ff8cca52d362e2793f293f5e1a52dd23152cbdda8b48faa3d46d0fa223b8a4c86c7459d82b5f2800753dc4a7ab6c321f7c3fb6cebc8a3cf4f001b82bf33b69d767dda1aa494adf171673bd222b4a8250f529392db306f803f56fcb24d618e7e006bc923097c6fbd8fcfb6e00c6e8ee585bb1ce6611f76e501673adb3a8d9cf00feb96e4c325d2184b44ae19d6741fa3fcbf1fe774d33d72d5cd0c6f4729b45c2cfe74738cf9a08cba1a5266f9c708aa3bfb697837fee870759a3b80d4a42bcbdf2f42e5d5156e5531b7cdc1d22bb5537f3e93c968ffee75a9eff4fc875cc784b468f5b6eb07d886274f2ca50c0617aeeab869919af4a3052d9bdd9d5d7aa9d3a55dcfb9dcbd3a815cc6bf7af5891b9238f7fd429eab4e2b04e69bc0143a0824523eab022effc7e9ce4b24e27df54ee8a2a4775e2af39f4f529cadaa6477b011c7b0af17bfff776f0bf46281b3bac742b0b8e8c41a8f9f486e1989558dbf6f96b80dbca24f62aa11dceb2d72b0d69d569ca0f708059223a3e62181991d5591c8379bc315f8d1683d350971bf84d4aea67a7dd39d2c2630076ae6f4bd2b8ae128290791c7f0e443c63f28b0522f3d30edce8474694c88973c0038c12b68369d3c7ce592ee55a27ee665a2c4f71093061fa8cfd911b7043233f052f435b892c97f957768a879590b1e8e732643fce86c6435d944d9036363dc3866101381c389d576bb66e192f6001101ff9b0a13a9e768bc8c8ec7a7d4af8b72152a5dffdefc0a30c24a0aba527bddb1ca69e4e478de97e2049102b2cbbaf00e29ce4e6ebeb4d26470ccfaf9655681d412ea2e41f4bd7fefb3effa82c05c0a65bb97076ca62165adeca283a5f3d4ae444da04b014e3aeb4ee9c717b8620f044d2e2c09141fc4d8d19f0cedb7fe12340cc55ccc7c91abf8e7398927273c9e91fc735ac7f360e2dca2f80171b1dd71b1bebed63f2768d53954528e16ffefcf582ff1be8a89d09bd102e4474c55e49411e7a0c67a1e69104a8a6fd7f2b17b9eed0da84896a178b23459ab08774d7a513b4890d1147fb8d011df3c480ae111512857a3ec6229b0a18905b3e0f26c4b14ca39d7e8bd8fdde035a91b2bcd920d9b3b007d2016563d75aa6ded7390a5e34ba1889959fe6de734e45810c8990de4b474b9a40aa82ec7dfc21ecc7f739e61ea746d7cdeb10a0cf54536757ebe5af4486ed6218e33ceadd5d5b423f7f12d13fb4b5187209e501896acd5b0b041aa597448457c042eabae95632455896be9ce468b63721914caf67744d395f07458ca3ef60a4e3550cd809eb3a1e70f3b9d1a7c851467a65cf9263e816c9eea2254b12fca25e122a9e34f5f33469f05acd73cff2f9560eff02e18cfcb312e0fe8adabeec07d1490e16040d0641c3cdd31c9473234627f52d8c526b27f6ba87cc8319848a91952f18c466a2605db8eb73448564c2ae6433ca5606b8aff46e307e221e7ffecba418eefdf714bcdac4559622cd900298a373bb68cfa6c8bc18735932bef13374374744b096f1ba31356f1c09a41c639a747129001980fb26eb961b80d609034ce2697a8b4c7581057bf84569d8ca49e37dda0c080c556219bbd2927513a7a12fd290db7b3930e7863a10137f7d651a8b0fade773996c163778e1a0610f346af66f329292ef78c66bc206d2b383edd5889cd3ffdc31e13b37f81ef87330e9c8f6bac867e4302a8745808fc134319b31599e9376ce0e2a782d64e285c6691486ceb0a56025b1a4162ee7df3d88e7601ff17fd171a0173d2eaac74965b84ffe7d89a5c9e709841da6b27e4aedf47eda9e470694aab69f3f4a7322625a9042e026cb2ae2c4061d69779cda51eeff713310a9c14b927dc74c6cf6f3a2de485bdcb69be3e587691a6bf1f2933d5c02c50d466b9321e4c7da49f0bd998c5a3a8befaaa02929ca041480154b8ebb25a4246d1ae3cd7c7e8c382af1455246f4c666b605cba7db76e547b897d202c3b98738d2995f7fd0ce2ff8e83302d32a6b3b463ed68ccd48624ee01e3c0d6c1640c3efe7ef0ae524241fb2a7df296f90d63a2f9d52948bea2d6bbd17d29e285a0018afb5aa98a83fdf1c8204a8f70b818daaa416d0db0e0ac357c631414586b7abd46f04298ef95e6d25bb97929231d801061fa9d11d3334e9c664120d6cce76dc5e059818da4b343565e9e4071ed4df9453a7d95cb8183c7c6b7d28a5f14328670b3c2c9a115263ebc7dffe035594be5e42fb184d05fd3709e8efe403e98c8e089fd105149b7262bc47f13e9bb6b4556db2587b7cb359106e9c2dfcb139bc16504998a64f9d0c463960222f3faada7110e8a1e057590aa1e8e4b018a6e646088b961bd947360368fa2f76d88017c0ca87a4926f2a88c2f98f26c09c2cccc451e3b1fdb09c10e91f6e1b4f49a7284ddd353a03a85492cf6d164f425f4024381f57983d10233cae0823b79d8d808388eeaa3f1d9eefd2dad9a7fef690b18419a505b494e65e949ff356a298e7211ce89df306a73116a99f28c7d046c083ef56ed3748ca781bbda4675a360d50a8c762a6fa5579635642a6e6b95dc868918d0a57980fe021084b7df37f4a6eaf78d8c8f69e6aafa58e0e0472a8f1ccf271c54f9d33e36bc2e315fc32f62c4804146bdf8119b006ff3f7a35b29091e6c60ef8ee8df243279277a33dfd5db91fcb4f582a347f471f0e7b4a623189515949d8166f87a0594e384b2feea11fbd55ae81409e67515333a431a098b1bf707316d91bf24585e2ba2391082d2bfe330fc97638b89803b2848f6598a8547749d23e6345d1d873b052d5fc02bef639df1ae8b062dbecf6ec2169b782f67ff8d0bc4072e3232f947e4a04b3937a7208649334b9ee6d7754d88db31d4a957b8c9982c50bc55fe4ed999b91bfe2354d2a7ad7ba0cc8d84903df903020eb1018b60715b6be7902828691fd4819205f052e47bd58478d81057aee7fd37b19a4efa1bd75f03a8dd59d9c81d9a9def013b5ee1392ceb51d07fb0d27adef823268101e29acbd23dedb804ecd9733aac8383eb149dca5b1e0c61ff13a50a7638a8e419ab4d77717009554174764922ccd50cefc5a2f01da8be96d4020e67bea4ed11bae9fbd7144556f957572ba245398b25d7e7383997b9e4a0f9f5a3815b4e211e50e961367e275f33d283c499719d364d814295b4520a64bacba85b4cedcddbf7f248119431076339beeaa03002e97751f3822df94cbc39819d568575c7ac14a783a3c8803c46cb8dc2269808b6b6c21139db93e37cca7de92d16cdea2bb06a6c2c12672e002162c926dceeb71b18f596e01a7781091cf256823dd2fe9c4f556c46114d74b4f89cc416f381d00e0a65a9d486478edc3b6d396a4166edf4c40ac0876a8525392f7f58d84731fa9f6f3182450720a9ea658350ec2b19687c86f9596b223bfd2b1d2c80f6104ce47731be624cced247c7452f9c25b8c8f90c6aed6f8e2d98a91587fad5c97cefa3c1b139bcbba4f60e22ec945f21545b37dd2fc3c0cf737f1eb10d8469ba9845daf49f07751b0978d6bd9543cc98dff52ed5a87535e04233d378a95edda2fc826a3080ba75da1b70167538ab747077a5e69b5564a56383f22288c847151b79c208a74e5a4d25ebc91615816fc60a5c8a65856be771dfa38ead3a6fb03a6bbaad68df77622ab30c59383bf1d12f00b7b6bac4d666b91695c66a146a43e40d15ebcb1a60e1e7b6e29c8b02e395fea18db801beec53b80e1298dda5f5f503cf753920d848797ee19be836819142a9b93a05400120cb31b833917dd05f7b2ff6acd7e79dbbdf79a7ee9fb02e0f7ad7d06d4796f82c1f697519d1a6c375267ff3d27e0ee7711ef0b2829ceb3c14380975ac7a59b02eb283f2ac09c695ca443077ab0d90c958074f6df6d211a779e50d8e186b8654ffb3d2e30f974c591f91476e56dbbb146766219c0991b08956f7287653e7158514f51132649228a1f30575f4e487d5f01aa1ec6c3986ca30fc831996b8c31f02ea16e5cc3be0814915f18d6643600dc0bf4794f866c0ca0a9ea02606866d8f5f913949bde564adfb15ad62abf3c1b118b117fee5f87d57e2f225b0ad83e2147e78678f72792ed28f251769fe71f608ff1168073b7d04c6856bd3f5eeebbe6bf7d2ad18ae924a94a2408b4b14d231d8e08dd0ba0730977105becc0b04673e4784fea0ff3f5006fce5596ce8c67a0910bb869064677bc2232ef3484e68ae9a224dfa98de8c1e70617b324e8c34cc6c6b4c5cee36f75aef550a27b710b5bfb2b772329805dfd63844ae0ade6474b37a75ed60fd8e9344ee38795290d3b971f1acfe339e4c79c6d8bc81b0b7428fdcd0c26cb4437c7a2ddb892fdf3afe4ae3f70ccf8a30cd5c2f3eff9a1c10a9e3e58cc0035b2abeb07149fe9e596a0920dfb98901c677046e9dbb1cc638ddc5fac8b52454d33f18bd3bd39bd45e0d51a2355a64a43507b9a9480c5b4944601eb1c90cfdc329eb93c5d7746c9de477d15592e5a76edd6a83294e2c112238f3a568435a01362b041b345c4de3d231f530585dfedb9bb0d0682185aeb0358325be15f24f8355326822d044966281b190f7f08ebdbe15a99b03b388dc42b4ba783912151432dfa34e3d76fc3658b66c4996cf60988040a551cc38b6f55ab1b43c1ea6f96290da4e1c20364bbbcd476f63e1bb5b06a25819b81a037b151ea624d477890b1075411f084f69a81602b03f322ae7bf6b1bbc043103cd9ab1d4a827245014fe1dac923382fbe185fb0a27c8463001dc33cd5eb653f6f97b3c5d69b959d92eb0a92ce09c7ae58b7812f01edf3102fd38ee65f3c8271253fee0b665cb7f2f86c7f20") print(res)