這篇文章主要介紹“Python如何實(shí)現(xiàn)自動(dòng)錄入ERP系統(tǒng)數(shù)據(jù)”,在日常操作中,相信很多人在Python如何實(shí)現(xiàn)自動(dòng)錄入ERP系統(tǒng)數(shù)據(jù)問(wèn)題上存在疑惑,小編查閱了各式資料,整理出簡(jiǎn)單好用的操作方法,希望對(duì)大家解答”Python如何實(shí)現(xiàn)自動(dòng)錄入ERP系統(tǒng)數(shù)據(jù)”的疑惑有所幫助!接下來(lái),請(qǐng)跟著小編一起來(lái)學(xué)習(xí)吧!
成都創(chuàng)新互聯(lián)2013年開(kāi)創(chuàng)至今,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項(xiàng)目成都做網(wǎng)站、網(wǎng)站制作、成都外貿(mào)網(wǎng)站建設(shè)網(wǎng)站策劃,項(xiàng)目實(shí)施與項(xiàng)目整合能力。我們以讓每一個(gè)夢(mèng)想脫穎而出為使命,1280元陸河做網(wǎng)站,已為上家服務(wù),為陸河各地企業(yè)和個(gè)人服務(wù),聯(lián)系電話:18980820575
軟件:Pycharm
環(huán)境: Python 3.7.9(考慮到客戶可能會(huì)有不同操作系統(tǒng),為了兼容性考慮)
技術(shù)庫(kù): requests、pandas、Pyqt5等(詳見(jiàn)依賴文件)
通過(guò)對(duì)客戶需求文檔分析和與溝通,大致有以下幾個(gè)需求:
根據(jù)“單號(hào)歸屬”批量向3個(gè)接口提交數(shù)據(jù)
需要一個(gè)GUI操作界面
支持不同的業(yè)務(wù)員登錄
總的來(lái)說(shuō)就是一個(gè)POST數(shù)據(jù)提交和GUI開(kāi)發(fā)。
這一塊主要用到的就是爬蟲(chóng)技術(shù),萬(wàn)年不變的步驟,都是先分析網(wǎng)頁(yè)。
通過(guò)抓包發(fā)現(xiàn),密碼是明文,難度就降低了一半,然后用正確的密碼再分析登錄成功后的返回。
def login(self, username: str, password: str): """ 登錄 """ url = "http://cloud.tiamaes.com:11349/erp/portal.bootstrap/SSOLoginAction/login.do" data = { "_tp_data": '{"parameters":{"userName":' + username + ',"pwd":' + password + '},"rowsets":{},"headers":{},"requestComponent":"0"}' } data = parse.urlencode(data).replace("+", "") resp = requests.post(url, headers=self.headers, data=data, verify=False) self.IDENTIFIER = resp.json()["headers"]["IDENTIFIER"] return self.IDENTIFIER
發(fā)現(xiàn)登錄成功后會(huì)返回一個(gè)“IDENTIFIER”參數(shù),值是加密字符串,這樣就很明顯,光看字面意思都知道這個(gè)肯定有用,所以先記錄下來(lái)。
由于我用的是測(cè)試賬號(hào),這個(gè)賬號(hào)提交的數(shù)據(jù)都要?jiǎng)h掉,為了不給別人注入太多的無(wú)效數(shù)據(jù),這里就不再實(shí)際錄入,以業(yè)務(wù)代碼來(lái)說(shuō)明。
獲取車輛信息
通過(guò)分析發(fā)現(xiàn),雖然客戶給了一部分車輛的信息,但是還有多缺失的信息,需要自己補(bǔ)充。通過(guò)抓包發(fā)現(xiàn),在輸入車輛編號(hào)以后,會(huì)發(fā)起一個(gè)Ajax請(qǐng)求,表單里其他信息就是Ajax請(qǐng)求返回的數(shù)據(jù)。
def get_car_details(self, car_no: str, IDENTIFIER: str): """ 獲取車輛信息 """ # print(self.IDENTIFIER) url = "http://cloud.tiamaes.com:11349/money/basis.inter/JwBusAction/getCacheJwBusByNo.do" data = { '_tp_data': '{"parameters": {"busNo": ' + str(car_no) + ', "dsName": "83"}, "rowsets": {}, "headers": {"IDENTIFIER": ' + IDENTIFIER + '}, "requestComponent": "0"}' } data = parse.urlencode(data).replace("+", "") resp = requests.post(url, headers=self.headers, data=data, verify=False) rows = resp.json()["rowsets"]["com.tp.basis.entity.entity.bus.BaJwBus"]["rows"][0] return rows
獲取人員信息
表單的人員信息我通過(guò)抓包沒(méi)有發(fā)現(xiàn),后來(lái)再一個(gè)頁(yè)面中找到了相關(guān)的數(shù)據(jù)。
這里稍微麻煩一點(diǎn),需要用正則把數(shù)據(jù)匹配出來(lái)。
def get_personal_info(self, IDENTIFIER: str):
"""
獲取個(gè)人信息
"""
url = "http://cloud.tiamaes.com:11349/money/money.action/CharteredAction/showDetail.do"
data = {
'_tp_data': '{"parameters":{"dsName":"83","method":"add","recId":"-1"},"rowsets":{},"headers":{"IDENTIFIER":' + IDENTIFIER + '},"requestComponent":"1"}'
}
data = parse.urlencode(data).replace("+", "")
resp = requests.post(url, headers=self.headers, data=data, verify=False)
json_data = eval(re.findall(r'.*?"rows":\[(.*?)\]', resp.text)[0])
return json_data
發(fā)起請(qǐng)求,提交數(shù)據(jù)
拿到了登錄返回的標(biāo)識(shí)符、車輛信息、人員信息,剩下的就是和客戶給的數(shù)據(jù)結(jié)合起來(lái),發(fā)起請(qǐng)求。需要注意的是,請(qǐng)求參數(shù)需要轉(zhuǎn)為url編碼,請(qǐng)求參數(shù)也是這個(gè)爬蟲(chóng)里面最麻煩的部分,這里給大家展示一個(gè)請(qǐng)求需要發(fā)送的參數(shù)。
參數(shù)多,格式要求也比較嚴(yán)格,整個(gè)開(kāi)發(fā)過(guò)程,這里調(diào)試花費(fèi)的時(shí)間也最長(zhǎng)。調(diào)試完正常應(yīng)該是把代碼簡(jiǎn)化一下,該合并的合并,我調(diào)試好了以后懶得再去改了,所以這一塊寫(xiě)的比較冗余。
def submit_data(self, i: dict, IDENTIFIER: str): """ 眾意數(shù)據(jù)提交 """ personal_info = self.get_personal_info(IDENTIFIER) # 獲取個(gè)人信息 personal_info_data = str(personal_info).replace("'", '"') # 將personal_info轉(zhuǎn)換為字符串 url = "http://cloud.tiamaes.com:11349/money/money.action/CharteredAction/saveForm.do" print(f'開(kāi)始處理--{i["單號(hào)歸屬"]}--數(shù)據(jù)') memo = f'工單號(hào){i["工單號(hào)"]}、餐費(fèi){i["餐費(fèi)"]}、住宿{i["住宿"]}、過(guò)路過(guò)橋費(fèi){i["過(guò)路過(guò)橋費(fèi)"]}、油費(fèi){i["油費(fèi)"]}、備注{i["備注"]}' # 拼接備注信息 car_infos = self.get_car_details(str(i["車號(hào)"]), IDENTIFIER) # 獲取車輛信息 pay_type = { "現(xiàn)金": "3", "轉(zhuǎn)賬": "2", "欠款": "1" } single_and_double = { "單程": "1", "雙程": "2" } colType = pay_type[i["結(jié)賬方式"]] # 獲取結(jié)賬方式編碼 oddEven = single_and_double[i["單雙程"]] # 獲取單雙程編碼 now_date = datetime.datetime.now().date().strftime("%Y-%m-%d") # 獲取當(dāng)前日期 .......(此處省略) data["_tp_data"] = data["_tp_data"].replace('"dsName":"83"', '"dsName":"82"') data = parse.urlencode(data).replace("+", "") # 將字典轉(zhuǎn)換成url編碼 resp = requests.post(url, headers=self.headers, data=data, verify=False).json() order_id = resp["rowsets"]["com.tp.money.entity.basic.Chartered"]["rows"][0]["recNo"] # 獲取訂單編號(hào) i["包車單號(hào)"] = order_id return data
gui開(kāi)發(fā)相對(duì)來(lái)說(shuō)比較簡(jiǎn)單,如果不想美化,Pyqt原生的插件就可以了,我這里是借用了上一個(gè)項(xiàng)目的經(jīng)驗(yàn),用僅有的知識(shí)做了一個(gè)無(wú)邊框界面和適當(dāng)?shù)拿阑?/p>
登錄
from PyQt5.QtCore import Qt from PyQt5.QtGui import QColor from PyQt5.QtWidgets import (QFrame, QMessageBox, QGraphicsDropShadowEffect) from Ui import login_ui from Ui.submit_ui_main import MySubmitForm from submit import TransitSubmit class MyLogin(login_ui.Ui_LoginForm, QFrame): def __init__(self, submit: TransitSubmit): super().__init__() # self.IDENTIFIER = None # self.my_main_window = None self.setupUi(self) self.submit = submit # 設(shè)置無(wú)邊框模式 self.setWindowFlag(Qt.FramelessWindowHint) # 將界面設(shè)置為無(wú)框 self.setAttribute(Qt.WA_TranslucentBackground) # 將界面屬性設(shè)置為半透明 self.shadow = QGraphicsDropShadowEffect() # 設(shè)定一個(gè)陰影,半徑為10,顏色為#444444,定位為0,0 self.shadow.setBlurRadius(10) self.shadow.setColor(QColor("#444444")) self.shadow.setOffset(0, 0) self.frame.setGraphicsEffect(self.shadow) # 為frame設(shè)定陰影效果 # ------------------------------------------------ self.show() self.pushButton_3.clicked.connect(self.close) # 關(guān)閉按鈕 self.pushButton_login.clicked.connect(self.do_login) # 登錄按鈕 # 以下是控制窗口移動(dòng)的代碼 def mousePressEvent(self, event): # 鼠標(biāo)左鍵按下時(shí)獲取鼠標(biāo)坐標(biāo),按下右鍵取消 if event.button() == Qt.LeftButton: self.m_flag = True self.m_Position = event.globalPos() - self.pos() event.accept() elif event.button() == Qt.RightButton: self.m_flag = False def mouseMoveEvent(self, QMouseEvent): # 鼠標(biāo)在按下左鍵的情況下移動(dòng)時(shí),根據(jù)坐標(biāo)移動(dòng)界面 if Qt.LeftButton and self.m_flag: self.move(QMouseEvent.globalPos() - self.m_Position) QMouseEvent.accept() def mouseReleaseEvent(self, QMouseEvent): # 鼠標(biāo)按鍵釋放時(shí),取消移動(dòng) self.m_flag = False # 登錄事件 def do_login(self): username = self.lineEdit_username.text() password = self.lineEdit_password.text() if not username or not password: QMessageBox.warning(self, '警告', '用戶名或密碼不能為空', QMessageBox.Yes) return else: IDENTIFIER = self.submit.login(username, password) if not IDENTIFIER: QMessageBox.warning(self, '警告', '用戶名或密碼錯(cuò)誤', QMessageBox.Yes) return self.hide() # 隱藏登錄界面 my_submit_form = MySubmitForm(self.submit, IDENTIFIER) my_submit_form.exec_() # 顯示主界面
業(yè)務(wù)操作
class MySubmitForm(submitform_ui.Ui_Dialog_Submit, QDialog): def __init__(self, submit: TransitSubmit, IDENTIFIER: str): super().__init__() ...... self.setupUi(self) ...... self.progressBar.hide() # 關(guān)閉進(jìn)度條顯示 self.setWindowFlags(Qt.FramelessWindowHint) # 無(wú)邊框 self.setAttribute(Qt.WA_TranslucentBackground) # 設(shè)置窗口透明 self.pushButton_mini.clicked.connect(self.showMinimized) # 實(shí)現(xiàn)最小化 self.pushButton_close.clicked.connect(self.close) # 實(shí)現(xiàn)關(guān)閉功能 ...... self.show() # 實(shí)現(xiàn)鼠標(biāo)拖拽功能 def mousePressEvent(self, event): self.pressX = event.x() # 記錄鼠標(biāo)按下的時(shí)候的坐標(biāo) self.pressY = event.y() def mouseMoveEvent(self, event): x = event.x() y = event.y() # 獲取移動(dòng)后的坐標(biāo) moveX = x - self.pressX moveY = y - self.pressY # 計(jì)算移動(dòng)了多少 positionX = self.frameGeometry().x() + moveX positionY = self.frameGeometry().y() + moveY # 計(jì)算移動(dòng)后主窗口在桌面的位置 self.move(positionX, positionY) # 移動(dòng)主窗口 ......
這里多說(shuō)一嘴,最開(kāi)始這里我用的和登錄一樣,使用的是QFrame,但是它沒(méi)有exec()方法,登錄成功后不能彈出,也可能是我知識(shí)有限,做不出來(lái)。通過(guò)分析源碼,發(fā)現(xiàn)QDialog有這個(gè)方法,可以實(shí)現(xiàn)彈出,后來(lái)又改了用QDialog做了一個(gè)無(wú)邊框界面。
剩下的打包就不多說(shuō)了,網(wǎng)上的教程很多,我這里用的是—D打包,用了upx壓縮,改了圖標(biāo),打包完整個(gè)項(xiàng)目有50多M。
到此,關(guān)于“Python如何實(shí)現(xiàn)自動(dòng)錄入ERP系統(tǒng)數(shù)據(jù)”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)?lái)更多實(shí)用的文章!