Software: Pycharm
Environment: Python 3.7.9(호환성 이유로 고객이 다른 운영 체제를 사용할 수 있다는 점 고려)
기술 라이브러리: 요청, pandas, Pyqt5 등 (자세한 내용은 종속성 파일 참조)
고객 요구 문서와의 분석 및 커뮤니케이션을 통해 대략 다음과 같은 요구 사항이 있습니다.
"odd"에 따라 일괄적으로 3개의 인터페이스에 데이터를 제출합니다. 번호 귀속"
GUI 운영 인터페이스 필요
다양한 영업사원의 로그인 지원
일반적으로 POST 데이터 제출 및 GUI 개발입니다.
이 섹션은 주로 크롤러 기술을 사용합니다. 수천년 동안 변함없이 유지되는 단계는 먼저 웹 페이지를 분석하는 것입니다.
패킷 캡처를 통해 비밀번호가 일반 텍스트라는 것을 발견하여 난이도를 절반으로 줄였습니다. 그런 다음 올바른 비밀번호를 사용하여 로그인 성공 후 반환을 분석합니다.
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
로그인에 성공하면 "IDENTIFIER" 매개변수가 반환된다는 사실을 발견했습니다. 이 값은 문자 그대로의 의미만 봐도 분명하므로 유용할 것입니다. 먼저 녹음해 보세요.
테스트 계정을 사용하고 있기 때문에 이 계정에서 제출한 데이터는 다른 사람에게 너무 많은 잘못된 데이터를 주입하지 않기 위해 여기서는 실제 입력이 이루어지지 않으며, 비즈니스 코드도 입력됩니다. 그것을 설명하는 데 사용됩니다.
차량 정보 가져오기
분석 결과, 고객님께서 차량 정보 중 일부를 제공하셨음에도 불구하고, 스스로 보완해야 할 부족한 정보가 여전히 많은 것으로 확인되었습니다. 패킷 캡처를 통해 차량 번호를 입력하면 Ajax 요청이 시작되고 양식의 다른 정보는 Ajax 요청에 의해 반환되는 데이터임을 확인했습니다.
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
인사정보 가져오기
패킷 캡쳐를 통해 폼에 있는 인사정보를 찾지 못했는데, 나중에 다른 페이지에서 관련 데이터를 발견했습니다.
이것은 좀 더 번거로운 일입니다. 데이터를 일치시키려면 정규 표현식을 사용해야 합니다.
def get_personal_info(self, IDENTIFIER: str): """ 获取个人信息 """ 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'<code>.*?"rows":\[(.*?)\]', resp.text)[0]) return json_data
요청 시작 및 데이터 제출
로그인에서 반환된 식별자, 차량 정보 및 개인 정보를 가져오고 나머지는 이를 고객이 제공한 데이터와 결합하여 요청을 시작하는 것입니다. 요청 매개변수는 URL 인코딩으로 변환되어야 한다는 점에 유의해야 합니다. 요청 매개변수는 이 크롤러에서 가장 까다로운 부분이기도 합니다. 여기서는 요청에서 전송해야 하는 매개변수를 보여줍니다.
매개변수가 많고 형식 요구 사항이 상대적으로 엄격합니다. 전체 개발 프로세스에서 여기에서 디버깅하는 데도 가장 오랜 시간이 걸립니다. 디버깅 후에는 코드를 단순화해야 합니다. 디버깅 후 병합해야 하는 병합을 변경하기에는 너무 게을러서 이 섹션이 상대적으로 중복됩니다.
def submit_data(self, i: dict, IDENTIFIER: str): """ 众意数据提交 """ personal_info = self.get_personal_info(IDENTIFIER) # 获取个人信息 personal_info_data = str(personal_info).replace("'", '"') # 将personal_info转换为字符串 url = "http://cloud.tiamaes.com:11349/money/money.action/CharteredAction/saveForm.do" print(f'开始处理--{i["单号归属"]}--数据') memo = f'工单号{i["工单号"]}、餐费{i["餐费"]}、住宿{i["住宿"]}、过路过桥费{i["过路过桥费"]}、油费{i["油费"]}、备注{i["备注"]}' # 拼接备注信息 car_infos = self.get_car_details(str(i["车号"]), IDENTIFIER) # 获取车辆信息 pay_type = { "现金": "3", "转账": "2", "欠款": "1" } single_and_double = { "单程": "1", "双程": "2" } colType = pay_type[i["结账方式"]] # 获取结账方式编码 oddEven = single_and_double[i["单双程"]] # 获取单双程编码 now_date = datetime.datetime.now().date().strftime("%Y-%m-%d") # 获取当前日期 .......(此处省略) data["_tp_data"] = data["_tp_data"].replace('"dsName":"83"', '"dsName":"82"') data = parse.urlencode(data).replace("+", "") # 将字典转换成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"] # 获取订单编号 i["包车单号"] = order_id return data
GUI 개발은 비교적 간단합니다. 꾸미고 싶지 않다면 Pyqt의 기본 플러그인을 사용하면 됩니다. 경계 없는 인터페이스와 적절한 미화.
Login
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 # 设置无边框模式 self.setWindowFlag(Qt.FramelessWindowHint) # 将界面设置为无框 self.setAttribute(Qt.WA_TranslucentBackground) # 将界面属性设置为半透明 self.shadow = QGraphicsDropShadowEffect() # 设定一个阴影,半径为10,颜色为#444444,定位为0,0 self.shadow.setBlurRadius(10) self.shadow.setColor(QColor("#444444")) self.shadow.setOffset(0, 0) self.frame.setGraphicsEffect(self.shadow) # 为frame设定阴影效果 # ------------------------------------------------ self.show() self.pushButton_3.clicked.connect(self.close) # 关闭按钮 self.pushButton_login.clicked.connect(self.do_login) # 登录按钮 # 以下是控制窗口移动的代码 def mousePressEvent(self, event): # 鼠标左键按下时获取鼠标坐标,按下右键取消 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): # 鼠标在按下左键的情况下移动时,根据坐标移动界面 if Qt.LeftButton and self.m_flag: self.move(QMouseEvent.globalPos() - self.m_Position) QMouseEvent.accept() def mouseReleaseEvent(self, QMouseEvent): # 鼠标按键释放时,取消移动 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, '警告', '用户名或密码错误', QMessageBox.Yes) return self.hide() # 隐藏登录界面 my_submit_form = MySubmitForm(self.submit, IDENTIFIER) my_submit_form.exec_() # 显示主界面
Business Operations
class MySubmitForm(submitform_ui.Ui_Dialog_Submit, QDialog): def __init__(self, submit: TransitSubmit, IDENTIFIER: str): super().__init__() ...... self.setupUi(self) ...... self.progressBar.hide() # 关闭进度条显示 self.setWindowFlags(Qt.FramelessWindowHint) # 无边框 self.setAttribute(Qt.WA_TranslucentBackground) # 设置窗口透明 self.pushButton_mini.clicked.connect(self.showMinimized) # 实现最小化 self.pushButton_close.clicked.connect(self.close) # 实现关闭功能 ...... self.show() # 实现鼠标拖拽功能 def mousePressEvent(self, event): self.pressX = event.x() # 记录鼠标按下的时候的坐标 self.pressY = event.y() def mouseMoveEvent(self, event): x = event.x() y = event.y() # 获取移动后的坐标 moveX = x - self.pressX moveY = y - self.pressY # 计算移动了多少 positionX = self.frameGeometry().x() + moveX positionY = self.frameGeometry().y() + moveY # 计算移动后主窗口在桌面的位置 self.move(positionX, positionY) # 移动主窗口 ......
여기서 좀 더 이야기해 보겠습니다. 처음에는 로그인과 동일한 QFrame을 사용했지만 exec()가 없습니다. 방법으로는 로그인 성공 후 팝업이 안 뜹니다. 제 지식이 부족해서 할 수 없는 것일 수도 있습니다. 소스 코드를 분석하여 QDialog에 팝업을 구현하는 방법이 있다는 것을 알았습니다. 나중에 이를 변경하고 QDialog를 사용하여 경계 없는 인터페이스를 만들었습니다.
나머지 패키징에 대해서는 많이 말하지 않겠습니다. 여기서 사용하는 것은 D 패키징, upx 압축 및 변경된 아이콘입니다.
위 내용은 Python에서 ERP 시스템 데이터를 자동으로 입력하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!