cbd_zhijian.py 66 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496
  1. from asyncio import ensure_future, shield
  2. from concurrent.futures import process
  3. from platform import python_build
  4. from turtle import st
  5. from PyQt5 import QtCore, QtGui, QtWidgets
  6. import shutil
  7. import os
  8. import sys
  9. import re
  10. import ast
  11. from urllib import parse
  12. import json
  13. import time
  14. import datetime
  15. import requests
  16. import pymongo
  17. import pymysql
  18. from xlrd import open_workbook
  19. from xlsxwriter.workbook import Workbook
  20. import numpy as np
  21. import cv2
  22. from aliyunsdkcore.vendored.requests.auth import HTTPBasicAuth
  23. import requests
  24. import datetime
  25. import time
  26. import hashlib
  27. from skimage import filters
  28. save_filename = ""
  29. pwd_str = "yf6021"
  30. toji_format = {
  31. 'font_name' : '宋体',
  32. 'font_size': 12,
  33. 'font_color': 'black',
  34. 'text_wrap': True,
  35. 'bold': False,
  36. 'fg_color': '92D050',
  37. 'align': 'center',
  38. 'valign': 'vcenter',
  39. 'border': 1,
  40. 'top': 1,
  41. 'left': 1,
  42. 'right': 1,
  43. 'bottom': 1
  44. }
  45. title_format = {
  46. 'font_name' : '宋体',
  47. 'font_size': 10,
  48. 'bold': True,
  49. 'align': 'center',
  50. 'valign': 'vcenter',
  51. "text_wrap": True,
  52. 'border': 1,
  53. 'top': 1,
  54. 'left': 1,
  55. 'right': 1,
  56. 'bottom': 1
  57. }
  58. merge_title_format = {
  59. 'font_name' : '宋体',
  60. 'font_size': 22,
  61. 'bold': True,
  62. 'align': 'center',
  63. 'valign': 'vcenter',
  64. "fg_color": "8DB4E2",
  65. 'border': 1,
  66. 'top': 1,
  67. 'left': 1,
  68. 'right': 1,
  69. 'bottom': 1
  70. }
  71. explain_formal = {
  72. 'font_name' : '宋体',
  73. 'font_size': 9,
  74. 'font_color': 'black',
  75. 'text_wrap': True,
  76. 'align': 'justify',
  77. 'valign': 'vcenter',
  78. 'border': 1,
  79. 'top': 1,
  80. 'left': 1,
  81. 'right': 1,
  82. 'bottom': 1
  83. }
  84. formal_format = {
  85. 'font_name' : '宋体',
  86. 'font_size': 9,
  87. 'font_color': 'black',
  88. 'fg_color': '77E88C',
  89. 'text_wrap': True,
  90. 'align': 'center',
  91. 'valign': 'vcenter',
  92. 'border': 1,
  93. 'top': 1,
  94. 'left': 1,
  95. 'right': 1,
  96. 'bottom': 1
  97. }
  98. common_format = {
  99. 'font_name' : '宋体',
  100. 'font_size': 9,
  101. 'font_color': 'black',
  102. "fg_color": 'E7EC73',
  103. 'text_wrap': True,
  104. 'align': 'center',
  105. 'valign': 'vcenter',
  106. 'border': 1,
  107. 'top': 1,
  108. 'left': 1,
  109. 'right': 1,
  110. 'bottom': 1
  111. }
  112. error_format = {
  113. 'font_name' : '宋体',
  114. 'font_size': 9,
  115. 'font_color': 'black',
  116. "fg_color": 'F4746A',
  117. 'text_wrap': True,
  118. 'align': 'center',
  119. 'valign': 'vcenter',
  120. 'border': 1,
  121. 'top': 1,
  122. 'left': 1,
  123. 'right': 1,
  124. 'bottom': 1
  125. }
  126. class Ui_MainWindow(object):
  127. """GUI界面"""
  128. def setupUi(self, MainWindow):
  129. MainWindow.setObjectName("MainWindow")
  130. MainWindow.resize(701, 644)
  131. font = QtGui.QFont()
  132. font.setFamily("Arial")
  133. font.setPointSize(12)
  134. MainWindow.setFont(font)
  135. icon = QtGui.QIcon("logo.ico")
  136. MainWindow.setWindowIcon(icon)
  137. self.centralwidget = QtWidgets.QWidget(MainWindow)
  138. self.centralwidget.setObjectName("centralwidget")
  139. self.pageTitleLabel = QtWidgets.QLabel(self.centralwidget)
  140. self.pageTitleLabel.setGeometry(QtCore.QRect(139, 50, 429, 42))
  141. font = QtGui.QFont()
  142. font.setFamily("楷体")
  143. font.setPointSize(28)
  144. self.pageTitleLabel.setFont(font)
  145. self.pageTitleLabel.setTextFormat(QtCore.Qt.AutoText)
  146. self.pageTitleLabel.setObjectName("pageTitleLabel")
  147. self.inputFileLabel = QtWidgets.QLabel(self.centralwidget)
  148. self.inputFileLabel.setGeometry(QtCore.QRect(180, 130, 121, 21))
  149. self.inputFileLabel.setObjectName("inputFileLabel")
  150. self.inputFileEdit = QtWidgets.QLineEdit(self.centralwidget)
  151. self.inputFileEdit.setGeometry(QtCore.QRect(310, 130, 151, 21))
  152. self.inputFileEdit.setFocusPolicy(QtCore.Qt.NoFocus)
  153. font = QtGui.QFont()
  154. font.setFamily("Arial")
  155. font.setPointSize(8)
  156. self.inputFileEdit.setFont(font)
  157. self.inputFileEdit.setObjectName("inputFileEdit")
  158. self.inputFileTool = QtWidgets.QToolButton(self.centralwidget)
  159. self.inputFileTool.setGeometry(QtCore.QRect(470, 130, 71, 21))
  160. font = QtGui.QFont()
  161. font.setFamily("Arial")
  162. font.setPointSize(10)
  163. self.inputFileTool.setFont(font)
  164. self.inputFileTool.setObjectName("inputFileTool")
  165. self.savePathLabel = QtWidgets.QLabel(self.centralwidget)
  166. self.savePathLabel.setGeometry(QtCore.QRect(180, 170, 121, 21))
  167. self.savePathLabel.setObjectName("savePathLabel")
  168. self.savePathEdit = QtWidgets.QLineEdit(self.centralwidget)
  169. self.savePathEdit.setGeometry(QtCore.QRect(310, 170, 151, 21))
  170. self.savePathEdit.setFocusPolicy(QtCore.Qt.NoFocus)
  171. font = QtGui.QFont()
  172. font.setFamily("Arial")
  173. font.setPointSize(8)
  174. self.savePathEdit.setFont(font)
  175. self.savePathEdit.setObjectName("savePathEdit")
  176. self.savePathTool = QtWidgets.QToolButton(self.centralwidget)
  177. self.savePathTool.setGeometry(QtCore.QRect(470, 170, 71, 21))
  178. font = QtGui.QFont()
  179. font.setFamily("Arial")
  180. font.setPointSize(10)
  181. self.savePathTool.setFont(font)
  182. self.savePathTool.setObjectName("savePathTool")
  183. self.startTimeLabel = QtWidgets.QLabel(self.centralwidget)
  184. self.startTimeLabel.setGeometry(QtCore.QRect(180, 210, 121, 21))
  185. self.startTimeLabel.setObjectName("startTimeLabel")
  186. self.startTimeEdit = QtWidgets.QDateTimeEdit(self.centralwidget)
  187. self.startTimeEdit.setGeometry(QtCore.QRect(310, 210, 151, 21))
  188. self.startTimeEdit.setMaximumDateTime(QtCore.QDateTime(QtCore.QDate(2028, 12, 31), QtCore.QTime(23, 59, 59)))
  189. self.startTimeEdit.setMinimumDateTime(QtCore.QDateTime(QtCore.QDate(2024, 1, 1), QtCore.QTime(12, 0, 0)))
  190. self.startTimeEdit.setObjectName("startTimeEdit")
  191. self.endTimeLabel = QtWidgets.QLabel(self.centralwidget)
  192. self.endTimeLabel.setGeometry(QtCore.QRect(180, 250, 121, 21))
  193. self.endTimeLabel.setObjectName("endTimeLabel")
  194. self.endTimeEdit = QtWidgets.QDateTimeEdit(self.centralwidget)
  195. self.endTimeEdit.setGeometry(QtCore.QRect(310, 250, 151, 21))
  196. self.endTimeEdit.setMaximumDateTime(QtCore.QDateTime(QtCore.QDate(2028, 12, 31), QtCore.QTime(23, 59, 59)))
  197. self.endTimeEdit.setMinimumDateTime(QtCore.QDateTime(QtCore.QDate(2024, 1, 1), QtCore.QTime(12, 0, 0)))
  198. self.endTimeEdit.setObjectName("endTimeEdit")
  199. self.platLabel = QtWidgets.QLabel(self.centralwidget)
  200. self.platLabel.setGeometry(QtCore.QRect(180, 290, 121, 21))
  201. self.platLabel.setObjectName("dianjiLabel")
  202. self.platBox = QtWidgets.QComboBox(self.centralwidget)
  203. self.platBox.setGeometry(QtCore.QRect(310, 290, 151, 21))
  204. self.platBox.addItems(['大数据平台','四情平台'])
  205. # 产品名称
  206. self.productLabel = QtWidgets.QLabel(self.centralwidget)
  207. self.productLabel.setGeometry(QtCore.QRect(180, 440, 121, 21))
  208. self.productLabel.setObjectName("productLabel")
  209. self.productBox = QtWidgets.QComboBox(self.centralwidget)
  210. self.productBox.setGeometry(QtCore.QRect(310, 440, 151, 21))
  211. self.productBox.addItems(['1.0测报灯','4.0测报灯', "水稻测报灯"])
  212. # 供电选择
  213. self.powerSupplyLabel = QtWidgets.QLabel(self.centralwidget)
  214. self.powerSupplyLabel.setGeometry(QtCore.QRect(180, 470, 121, 21))
  215. self.powerSupplyLabel.setObjectName("powerSupplyLabel")
  216. self.powerSupplyBox = QtWidgets.QComboBox(self.centralwidget)
  217. self.powerSupplyBox.setGeometry(QtCore.QRect(310, 470, 151, 21))
  218. self.powerSupplyBox.addItems(['AC','DC'])
  219. font = QtGui.QFont()
  220. font.setFamily("Arial")
  221. font.setPointSize(11)
  222. self.platBox.setFont(font)
  223. self.platBox.setObjectName("platBox")
  224. self.stm8vsLabel = QtWidgets.QLabel(self.centralwidget)
  225. self.stm8vsLabel.setGeometry(QtCore.QRect(180, 330, 121, 21))
  226. self.stm8vsLabel.setObjectName("stm8vsLabel")
  227. self.stm8vsEdit = QtWidgets.QLineEdit(self.centralwidget)
  228. self.stm8vsEdit.setGeometry(QtCore.QRect(310, 330, 151, 21))
  229. font = QtGui.QFont()
  230. font.setFamily("Arial")
  231. font.setPointSize(11)
  232. self.stm8vsEdit.setFont(font)
  233. self.stm8vsEdit.setObjectName("stm8vsEdit")
  234. self.dverLabel = QtWidgets.QLabel(self.centralwidget)
  235. self.dverLabel.setGeometry(QtCore.QRect(180, 370, 121, 21))
  236. self.dverLabel.setObjectName("dverLabel")
  237. self.dverEdit = QtWidgets.QLineEdit(self.centralwidget)
  238. self.dverEdit.setGeometry(QtCore.QRect(310, 370, 151, 21))
  239. font = QtGui.QFont()
  240. font.setFamily("Arial")
  241. font.setPointSize(11)
  242. self.dverEdit.setFont(font)
  243. self.dverEdit.setObjectName("dverEdit")
  244. self.orderLabel = QtWidgets.QLabel(self.centralwidget)
  245. self.orderLabel.setGeometry(QtCore.QRect(180, 410, 121, 21))
  246. self.orderLabel.setObjectName("dverLabel")
  247. self.orderEdit = QtWidgets.QLineEdit(self.centralwidget)
  248. self.orderEdit.setGeometry(QtCore.QRect(310, 410, 151, 21))
  249. # 进度条与按钮
  250. font = QtGui.QFont()
  251. font.setFamily("Arial")
  252. font.setPointSize(11)
  253. self.orderEdit.setFont(font)
  254. self.orderEdit.setObjectName("dverEdit")
  255. self.pushButton = QtWidgets.QPushButton(self.centralwidget)
  256. self.pushButton.setGeometry(QtCore.QRect(300, 520, 111, 41))
  257. self.pushButton.setObjectName("pushButton")
  258. self.progressBar = QtWidgets.QProgressBar(self.centralwidget)
  259. self.progressBar.setGeometry(QtCore.QRect(130, 540, 491, 31))
  260. self.progressBar.setProperty("value", 0)
  261. self.progressBar.setVisible(False)
  262. self.progressBar.setObjectName("progressBar")
  263. MainWindow.setCentralWidget(self.centralwidget)
  264. self.retranslateUi(MainWindow)
  265. self.inputFileTool.clicked.connect(self.input_file_path)
  266. self.savePathTool.clicked.connect(self.out_save_location)
  267. self.pushButton.clicked.connect(self.on_click)
  268. QtCore.QMetaObject.connectSlotsByName(MainWindow)
  269. def retranslateUi(self, MainWindow):
  270. _translate = QtCore.QCoreApplication.translate
  271. MainWindow.setWindowTitle(_translate("MainWindow", "杀虫灯质检工具"))
  272. self.savePathTool.setText(_translate("MainWindow", "选择文件夹"))
  273. self.inputFileTool.setText(_translate("MainWindow", "选择文件"))
  274. self.pageTitleLabel.setText(_translate("MainWindow", "<html><head/><body><p><span style=\" font-weight:600; color:#3a45aa;\">云飞测报灯设备质检工具</span></p></body></html>"))
  275. self.savePathLabel.setText(_translate("MainWindow", "|输出文件位置:"))
  276. self.inputFileLabel.setText(_translate("MainWindow", "|输入文件位置:"))
  277. self.startTimeLabel.setText(_translate("MainWindow", "|开始时间:"))
  278. self.endTimeLabel.setText(_translate("MainWindow", "|结束时间:"))
  279. self.platLabel.setText(_translate("MainWindow", "|检验平台:"))
  280. self.productLabel.setText(_translate("MainWindow", "|产品名称:"))
  281. self.powerSupplyLabel.setText(_translate("MainWindow", "|供电选择:"))
  282. self.stm8vsLabel.setText(_translate("MainWindow", "|系统固件版本号:"))
  283. self.dverLabel.setText(_translate("MainWindow", "|RTU固件版本号: "))
  284. self.orderLabel.setText(_translate("MainWindow", "|任务单号:"))
  285. self.pushButton.setText(_translate("MainWindow", "开始导出"))
  286. def input_file_path(self):
  287. file_path = QtWidgets.QFileDialog.getOpenFileName(None,"选取文件","./","All Files (*.xlsx;*.xls);;Text Files (*.txt)")
  288. self.inputFileEdit.setText(file_path[0])
  289. self.inputFileEdit.setStyleSheet("color:black;")
  290. def out_save_location(self):
  291. fname = QtWidgets.QFileDialog.getExistingDirectory(None, '选取文件夹', './')
  292. self.savePathEdit.setText(fname)
  293. self.savePathEdit.setStyleSheet("color:black;")
  294. def on_click(self):
  295. file_path = self.inputFileEdit.text()
  296. save_path = self.savePathEdit.text()
  297. start_time = self.startTimeEdit.dateTime().toPyDateTime()
  298. end_time = self.endTimeEdit.dateTime().toPyDateTime()
  299. set_plat = self.platBox.currentText()
  300. set_stm8vs = self.stm8vsEdit.text()
  301. set_dver = self.dverEdit.text()
  302. set_order = self.orderEdit.text()
  303. set_product = self.productBox.currentText()
  304. set_power = self.powerSupplyBox.currentText()
  305. if not all ([file_path,save_path,start_time,end_time,set_stm8vs,set_dver,set_order]):
  306. QtWidgets.QMessageBox.information(None, "提示", "选项未填写完整")
  307. elif end_time < start_time:
  308. QtWidgets.QMessageBox.information(None, "提示", "结束日期应在开始日期之后")
  309. else:
  310. xls = open_workbook(file_path)
  311. sheet_object = xls.sheets()[0]
  312. ncols = sheet_object.ncols
  313. read_dict = {}
  314. for i in range(ncols):
  315. col_value = sheet_object.col_values(i)
  316. read_dict[col_value[0]] = col_value[1:]
  317. device_list = read_dict.get("设备ID")
  318. if device_list:
  319. self.inputFileTool.setEnabled(False)
  320. self.savePathTool.setEnabled(False)
  321. self.startTimeEdit.setEnabled(False)
  322. self.endTimeEdit.setEnabled(False)
  323. self.platBox.setEnabled(False)
  324. self.productBox.setEnabled(False)
  325. self.powerSupplyBox.setEnabled(False)
  326. self.stm8vsEdit.setEnabled(False)
  327. self.dverEdit.setEnabled(False)
  328. self.orderEdit.setEnabled(False)
  329. self.pushButton.setEnabled(False)
  330. self.pushButton.setText("执行中...")
  331. self.runThread = CBDThread(device_list,save_path,start_time,end_time,set_plat,set_stm8vs,set_dver,set_order, set_product, set_power)
  332. self.runThread.proess_signal.connect(self.set_progressbar_value)
  333. self.runThread.start()
  334. self.progressBar.setVisible(True)
  335. else:
  336. QtWidgets.QMessageBox.information(None, "提示", "输入文件无'设备ID'列或该列无数据")
  337. def set_progressbar_value(self, value):
  338. self.progressBar.setValue(value)
  339. if value == 100:
  340. QtWidgets.QMessageBox.information(None, "提示", "文件导出完毕!导出文件名:\n{}".format(save_filename))
  341. self.inputFileTool.setEnabled(True)
  342. self.inputFileEdit.setText("")
  343. self.savePathTool.setEnabled(True)
  344. self.savePathEdit.setText("")
  345. self.startTimeEdit.setEnabled(True)
  346. self.endTimeEdit.setEnabled(True)
  347. self.platBox.setEnabled(True)
  348. self.productBox.setEnabled(True)
  349. self.powerSupplyBox.setEnabled(True)
  350. self.stm8vsEdit.setEnabled(True)
  351. self.stm8vsEdit.setText("")
  352. self.dverEdit.setEnabled(True)
  353. self.dverEdit.setText("")
  354. self.orderEdit.setEnabled(True)
  355. self.orderEdit.setText("")
  356. self.pushButton.setEnabled(True)
  357. self.pushButton.setText("开始导出")
  358. self.progressBar.setVisible(False)
  359. self.progressBar.setValue(0)
  360. return
  361. class CBDThread(QtCore.QThread):
  362. """涉及进度条需主界面动态执行GUI,线程执行业务逻辑,避免主页面卡死"""
  363. proess_signal = QtCore.pyqtSignal(int)
  364. def __init__(self,device_list,save_path,start_time,end_time,set_plat,set_stm8vs,set_dver,set_order, set_product, set_power):
  365. super(CBDThread, self).__init__()
  366. self.device_list = device_list
  367. self.save_path = save_path
  368. self.start_time = time.mktime(start_time.timetuple())
  369. self.end_time = time.mktime(end_time.timetuple())
  370. # self.start_time = 1689737400
  371. # self.end_time = 1689753600
  372. self.start_time_str = start_time.strftime("%y-%m-%d %H:%M:%S")
  373. self.end_time_str = end_time.strftime("%y-%m-%d %H:%M:%S")
  374. self.set_plat = set_plat
  375. self.set_stm8vs = set_stm8vs
  376. self.set_dver = set_dver
  377. self.set_order = set_order
  378. self.set_product = set_product
  379. self.set_power = set_power
  380. self.user = parse.quote_plus("root")
  381. self.passwd = parse.quote_plus("yfkj@6020")
  382. self.myclient = pymongo.MongoClient("mongodb://{0}:{1}@8.136.98.49:57017/".format(self.user,self.passwd))
  383. self.db = self.myclient.smartfarming
  384. self.device_collection = self.db.sa_device
  385. self.cbd_collection = self.db.sa_device_cbd_data
  386. self.cbd_photo = self.db.sa_device_cbdphoto
  387. self.cbd_alerm = self.db.sa_alarm_record
  388. self.device_config = self.db.sa_device_config
  389. self.config = {
  390. 'host': '120.27.222.26',
  391. 'port': 3306,
  392. 'user': 'yfwlw',
  393. 'password': 'sql_yfkj_6019',
  394. 'db': 'yfwlw',
  395. 'charset': 'utf8mb4',
  396. 'cursorclass': pymysql.cursors.DictCursor,
  397. }
  398. self.connection = pymysql.connect(**self.config)
  399. self.cursor = self.connection.cursor()
  400. def get_position(self, device_id):
  401. device = self.device_collection.find_one({"device_id": device_id})
  402. province, city, district = device.get("province"), device.get("city"), device.get("district")
  403. position_device = f"{province}{city}{district}"
  404. if position_device:
  405. return position_device
  406. lng, lat = device.get("lng"), device.get("lat")
  407. if lng and lat:
  408. try:
  409. ret = requests.post("http://api.map.baidu.com/geocoder?location=%s,%s&coord_type=gcj02&output=json"%(lat,lng))
  410. ret_json = json.loads(ret.text)
  411. province, city, district = ret_json["result"]["addressComponent"]["province"], \
  412. ret_json["result"]["addressComponent"]["city"], \
  413. ret_json["result"]["addressComponent"]["district"]
  414. return province + city + district
  415. except Exception as e:
  416. return False
  417. else:
  418. return False
  419. def sim_info(self,iccid):
  420. # 时间戳 用于获取sign
  421. timestamp = int(time.time())
  422. current_milli_time = lambda: int(round(time.time() * 1000))
  423. data_1 = "appid=%s&iccid=%s&timestamp=%s%s"%("102420177762",iccid,current_milli_time(),"6397d7e6a56589f1d93284e9800493e1")
  424. sign = hashlib.sha256(data_1.encode('utf-8')).hexdigest()
  425. data = {"appid": "102420177762", "iccid": iccid, "timestamp":current_milli_time(),"sign":sign}
  426. url = "https://api.simboss.com/2.0/device/detail"
  427. try:
  428. status = 1
  429. ret = requests.post(url, data=data)
  430. code = json.loads(ret.text)["code"]
  431. if code == "0":
  432. status = 1
  433. else:
  434. url = 'http://sim.brlink.cn/api/open/iotcard/card'
  435. appkey = "iaO2DKgS8KdlnVgU"
  436. appsecret = "qzKgO4sBdzMrjRwv9H22S9ufepNv8Hl5ehPqkYVD31DCICjyKwqUdj7zihQQKfgx"
  437. status = 2
  438. ret = requests.post(url,json={'iccid':iccid},auth=HTTPBasicAuth(appkey,appsecret),timeout=(5,10))
  439. codes = json.loads(ret.text)["code"]
  440. if codes == 0:
  441. status = 2
  442. else:
  443. url = "https://jsnl.xmnengjia.com/open/api/module/cards"
  444. data = {"iccids":[iccid]}
  445. data = json.dumps(data)
  446. ret = requests.post(url,data=data,timeout=(10,30))
  447. status = 3
  448. except:
  449. status = 0
  450. ret = 0
  451. day = 0
  452. if ret:
  453. try:
  454. result = json.loads(ret.text)
  455. expiry_date = result.get("data", {}).get("expiry_date")
  456. now_date = int(time.time())
  457. time_difference = int((expiry_date - now_date) / 3600 / 24)
  458. if time_difference < 30:
  459. return [1,"有效期剩余{}天".format(time_difference)]
  460. elif time_difference >= 181:
  461. return [1,"有效期剩余{}天".format(time_difference)]
  462. else:
  463. return [1,"有效期剩余{}天".format(time_difference)]
  464. except:
  465. return [1, "查询无结果"]
  466. else:
  467. return [1, "查询无结果"]
  468. def _imageToMatrix(self, image):
  469. """
  470. 根据名称读取图片对象转化矩阵
  471. :param strName:
  472. :return: 返回矩阵
  473. """
  474. imgMat = np.matrix(image)
  475. return imgMat
  476. def preImgOps(self, img):
  477. """
  478. 图像的预处理操作
  479. :param img: 图像的而明朝
  480. :return: 灰度化和resize之后的图片对象
  481. """
  482. # 预处理操作
  483. try:
  484. reImg = cv2.resize(img, (300, 400), interpolation=cv2.INTER_CUBIC) #
  485. img2gray = cv2.cvtColor(reImg, cv2.COLOR_BGR2GRAY) # 将图片压缩为单通道的灰度图
  486. return reImg,img2gray, True
  487. except Exception as e:
  488. return None, None, False
  489. def ReadImage(self, im_file):
  490. """
  491. 读取图片
  492. :param im_file: 图像的路径
  493. :return: 利用opencv读取的完整图像img 托虫板内接四边形图像cutimg 内接四边形的坐标c[xmin,ymin,xmax,ymax]
  494. """
  495. try:
  496. img = cv2.imdecode(np.fromfile(im_file, dtype=np.uint8), cv2.IMREAD_COLOR)
  497. ImgShape = img.shape
  498. center = [ImgShape[1]/2,ImgShape[0]/2] #圆盘中心
  499. radius = int(ImgShape[0]/2) #圆盘半径
  500. length = pow(2,0.5)*radius #圆的内接正方形边长
  501. Size = int(length/2)
  502. xmin = int(center[0] - Size)
  503. xmax = int(center[0] + Size)
  504. ymin = int((center[1] - Size)*1.1)
  505. ymax = int(center[1] + Size)
  506. c = [xmin,ymin,xmax,ymax]
  507. cutimg = img[ymin:ymax,xmin:xmax]
  508. return img,cutimg,c, True
  509. except Exception as e:
  510. return None, None, None, False
  511. def getFileName(self, imageFile):
  512. url, FileName = os.path.split(imageFile)
  513. tamp = []
  514. tamp.append(FileName)
  515. return tamp
  516. #图像清晰度检测
  517. def predict(self, imageFile):
  518. """
  519. 图像清晰度检测
  520. Args:
  521. imageFile(dir or file):单张图片或者存放图片的文件夹
  522. """
  523. result = [] #存放清晰度评分
  524. imgList = [] #存放识别后图片
  525. if os.path.isdir(imageFile):
  526. imageFileList = [os.path.join(imageFile,x) for x in os.listdir(imageFile)]
  527. for image in imageFileList:
  528. img,cutimg,c, is_get = self.ReadImage(image)
  529. if is_get:
  530. reImg, img2gray, is_imgs= self.preImgOps(cutimg)
  531. if is_imgs:
  532. f = self._imageToMatrix(img2gray)
  533. tmp = filters.sobel(f)
  534. source=np.sum(tmp**2)
  535. source=np.sqrt(source)
  536. result.append(source)
  537. imgList.append(img)
  538. elif os.path.isfile(imageFile):
  539. img,cutimg,c, is_get = self.ReadImage(imageFile)
  540. if is_get:
  541. reImg, img2gray, is_imgs= self.preImgOps(cutimg)
  542. if is_imgs:
  543. f = self._imageToMatrix(img2gray)
  544. tmp = filters.sobel(f)
  545. source=np.sum(tmp**2)
  546. source=np.sqrt(source)
  547. result.append(source)
  548. imgList.append(img)
  549. return result
  550. def __bigdata_iamge_verify(self, device_id):
  551. # 对图片质量进行打分并求平均值
  552. scores_time = {}
  553. photo_obj = self.cbd_photo.find({"device_id":str(device_id), 'addtime': {"$gte": int(self.start_time) - 3600 * 4,"$lte":self.start_time}})
  554. for p in photo_obj:
  555. photo_addr = p.get("addr")
  556. addtime = p.get("addtime")
  557. # 把图片下载到本地
  558. local_dir = "org_image"
  559. file_name = photo_addr.split("/")[-1]
  560. file_name = file_name.replace("?", "")
  561. os.makedirs(local_dir) if not os.path.exists(local_dir) else None
  562. print(photo_addr, "----------img----------")
  563. # response = requests.get(f"https://bigdata-image.oss-cn-hangzhou.aliyuncs.com/Basics/cbd/{photo_addr}")
  564. if "ftp" in photo_addr:
  565. response = requests.get(photo_addr)
  566. else:
  567. response = requests.get(f"http://8.136.98.49:8003/Basics/cbd/{photo_addr}")
  568. with open(os.path.join(local_dir, file_name), 'wb') as f:
  569. f.write(response.content)
  570. result_score = self.predict(os.path.join(local_dir, file_name))
  571. print(result_score)
  572. if result_score:
  573. avg_score = sum(result_score)/len(result_score)
  574. scores_time[str(addtime)] = avg_score
  575. try:
  576. os.remove(os.path.join(local_dir, file_name))
  577. except:
  578. pass
  579. try:
  580. shutil.rmtree(local_dir)
  581. except:
  582. pass
  583. print(scores_time, "----------------------")
  584. return scores_time
  585. def __siqing_image_verify(self, device_id):
  586. # 对图片质量进行打分并求平均值
  587. scores_time = {}
  588. sql = "SELECT * FROM AppInfoManage_cbdphoto WHERE equip_id_id='{}' AND upl_time BETWEEN '{}' AND '{}' order by 'alarm_time';".format(device_id,self.start_time_str,self.end_time_str)
  589. self.cursor.execute(sql)
  590. photo_obj = self.cursor.fetchall()
  591. for p in photo_obj:
  592. photo_addr = p.get("addr")
  593. addtime = p.get("addtime")
  594. # 把图片下载到本地
  595. local_dir = "org_image"
  596. file_name = photo_addr.split("/")[-1]
  597. os.makedirs(local_dir) if not os.path.exists(local_dir) else None
  598. response = requests.get(photo_addr)
  599. with open(os.path.join(local_dir, file_name), 'wb') as f:
  600. f.write(response.content)
  601. result_score = self.predict(os.path.join(local_dir, file_name))
  602. avg_score = sum(result_score)/len(result_score)
  603. scores_time[str(addtime)] = avg_score
  604. try:
  605. os.remove(os.path.join(local_dir, file_name))
  606. except:
  607. pass
  608. return scores_time
  609. def mongo_ping(self):
  610. """mongo-ping预防连接失效"""
  611. try:
  612. self.myclient.admin.command('ping')
  613. except: # "ConnectionFailure"
  614. self.myclient = pymongo.MongoClient("mongodb://{0}:{1}@8.136.98.49:57017/".format(self.user,self.passwd))
  615. self.db = self.myclient.smartfarming
  616. self.device_collection = self.db.sa_device
  617. self.cbd_collection = self.db.sa_device_cbd_data
  618. self.cbd_photo = self.db.sa_device_cbdphoto
  619. self.cbd_alerm = self.db.sa_alarm_record
  620. def sql_ping(self):
  621. """mysql-ping 预防连接失效"""
  622. try:
  623. self.connection.ping()
  624. except:
  625. self.connection = pymysql.connect(**self.config)
  626. self.cursor = self.connection.cursor()
  627. def __time_dif(self,checkdatetime):
  628. """计算时间差"""
  629. nowdatetime = datetime.datetime.now()
  630. checkdatetime = datetime.datetime.strptime(checkdatetime, "%Y-%m-%d %H:%M:%S")
  631. timedif = checkdatetime - nowdatetime
  632. return timedif.days
  633. def device_their_platform(self,shortId):
  634. """确定设备所在平台已经完整设备号"""
  635. self.mongo_ping()
  636. self.sql_ping()
  637. regex = re.compile('.*{}$'.format(shortId))
  638. bd_device_dict = self.device_collection.find_one(
  639. filter = {"device_id":regex,"device_type_id":3},
  640. projection = {'_id': 0},
  641. sort = [('uptime', pymongo.DESCENDING)]
  642. )
  643. device_sql = "SELECT * FROM AppInfoManage_cbdstatus WHERE equip_id_id LIKE '%{}' ORDER BY upl_time DESC LIMIT 1;".format(shortId)
  644. self.cursor.execute(device_sql)
  645. sq_device_dict = self.cursor.fetchone()
  646. if bd_device_dict and sq_device_dict:
  647. bd_upltime = bd_device_dict["uptime"]
  648. sq_upltime = time.mktime(sq_device_dict["upl_time"].timetuple())
  649. if bd_upltime >= sq_upltime:
  650. d_id = bd_device_dict["id"]
  651. deviceId = bd_device_dict["device_id"]
  652. platform = "大数据平台"
  653. else:
  654. d_id = ""
  655. deviceId = sq_device_dict["equip_id_id"]
  656. platform = "四情平台"
  657. elif bd_device_dict and not sq_device_dict:
  658. d_id = bd_device_dict["id"]
  659. deviceId = bd_device_dict["device_id"]
  660. platform = "大数据平台"
  661. return d_id,deviceId,platform
  662. elif not bd_device_dict and sq_device_dict:
  663. d_id = ""
  664. deviceId = sq_device_dict["equip_id_id"]
  665. platform = "四情平台"
  666. else:
  667. d_id = ""
  668. deviceId = "平台无此设备"
  669. platform = "未知"
  670. return d_id,deviceId,platform
  671. def eliminate_adjacent_duplicates(self, string):
  672. result = ''
  673. prev_char = None
  674. for char in string:
  675. if char != prev_char:
  676. result += char
  677. prev_char = char
  678. return result
  679. def __bigdata_verify(self,data_cursor,data_status,device_id_info, image_score, proess, data_temps, position, tt):
  680. """
  681. 获取大数据平台查询结果
  682. 1. 合格
  683. 2. 不合格
  684. """
  685. current_jg = [1,""] # 振动电机工作状态下电流匹配度
  686. up_jg = [1,"无电流不足信息"] # 上仓门电机电机工作状态
  687. down_jg = [1,"无电流不足信息"] # 下仓门电机电机工作状态
  688. lig_jg = [1,"无电流不足信息"] # 灯管工作状态下
  689. machine_jg = [1, "无电流不足信息"]# 清扫/转盘电机工作状态
  690. camera_jg = [1, "无电流不足信息"] # 相机工作状态
  691. current_jg_ct, up_jg_ct, down_jg_ct, lig_jg_ct, camera_jg_ct, machine_ct = 0,0,0,0,0,0
  692. for alerm in data_status:
  693. alarm_time = datetime.datetime.fromtimestamp(alerm.get("alarm_time")).strftime("%Y-%m-%d %H:%M:%S")
  694. a = alerm.get("alarm_desc", "")
  695. if a:
  696. a = json.loads(a)
  697. if a.get("type", "") == "震动电机":
  698. if "电流不足" in a.get("status"):
  699. current_jg_ct += 1
  700. elif a.get("type", "") == "上仓门电机":
  701. if "电流不足" in a.get("status"):
  702. up_jg_ct += 1
  703. elif a.get("type", "") == "下仓门电机":
  704. if "电流不足" in a.get("status"):
  705. down_jg_ct += 1
  706. elif a.get("type", "") == "灯管":
  707. if "电流不足" in a.get("status"):
  708. lig_jg_ct += 1
  709. elif a.get("type", "") in ["清扫电机", "转盘电机"]:
  710. if "电流不足" in a.get("status"):
  711. machine_ct += 1
  712. elif a.get("type", "") == "相机":
  713. print(a, "电流不足", alarm_time)
  714. if "电流不足" in a.get("status"):
  715. camera_jg_ct += 1
  716. # if current_jg_ct:
  717. # current_jg = [1,""]
  718. # if up_jg_ct:
  719. # up_jg = [0,f"有电流不足信息{str(up_jg_ct)}条"]
  720. # if down_jg_ct:
  721. # down_jg = [0,f"有电流不足信息{str(down_jg_ct)}条"]
  722. # if lig_jg_ct:
  723. # lig_jg = [0,f"有电流不足信息{str(lig_jg_ct)}条"]
  724. # if camera_jg_ct:
  725. # camera_jg = [0,f"有电流不足信息{str(camera_jg_ct)}条"]
  726. # if machine_ct:
  727. # machine_jg = [0,f"有电流不足信息{str(machine_ct)}条"]
  728. iccid = ""
  729. lng_lst = []
  730. lat_lst = []
  731. work_status = [] # 工作状态切换
  732. elec_vs_lst = [] # 电路板固件版本号
  733. v_vs_lst = [] # 电压
  734. tr_k_jg_lst = [] # 温控
  735. rain_k_jg_lst = []# 雨控
  736. infor_jg_count = 0 # 上报信息条数
  737. heat_temp_lst = []# 加热仓温度
  738. temp_hum = [] # 环境温度 环境湿度
  739. total_time = [] # 总时长
  740. start_time = 0
  741. information_count = [] # 上报时间
  742. infor_lamp = {}
  743. dver_lst = []
  744. for c in data_temps:
  745. device_c = c.get("device_data", "")
  746. if device_c:
  747. device_c = ast.literal_eval(device_c)
  748. tp = device_c.get("tps", "")
  749. if tp == "1":
  750. tr_k_jg_lst.append(tp)
  751. for index, cur in enumerate(data_cursor):
  752. start_time = int(cur.get("addtime", 0))
  753. device_data = cur.get("device_data", "")
  754. if device_data:
  755. device_data = ast.literal_eval(device_data)
  756. iccid = device_data.get("iccid", "")
  757. lamp = device_data.get("lamp")
  758. if lamp == "0":
  759. infor_lamp[str(start_time)] = lamp
  760. information_count.append(start_time)
  761. lng = device_data.get("lng", "")
  762. lat = device_data.get("lat", "")
  763. try:
  764. lng_lst.append(float(lng))
  765. lat_lst.append(float(lat))
  766. except Exception as e:
  767. lng_lst.append(0)
  768. lat_lst.append(0)
  769. work_status.append(device_data.get("ws", ""))
  770. elec_vs_lst.append(device_data.get("dver", ""))
  771. v_vs_lst.append(float(device_data.get("vbat", 0)))
  772. rain_k_jg_lst.append(device_data.get("rps", ""))
  773. infor_jg_count += 1
  774. heat_temp_lst.append(float(device_data.get("hrt", 0)))
  775. temp_hum.append({"at": device_data.get("at", ""), "ah": device_data.get("ah", "")})
  776. if self.set_power == "AC":
  777. total_time.append(int(device_data.get("tt")))
  778. elif self.set_power == "DC":
  779. st = device_data.get("st") # 19:00
  780. et = device_data.get("et") # 08:00
  781. st = int(st.split(":")[0])
  782. et = int(et.split(":")[0])
  783. if st > et:
  784. total = 24 - st + et
  785. else:
  786. total = et - st
  787. total_time.append(total)
  788. dver = device_data.get("dver", "")
  789. if dver:
  790. dver_lst.append(dver)
  791. # 检查卡号及有效期
  792. if self.set_product != "水稻测报灯":
  793. card_jg = self.sim_info(iccid)
  794. else:
  795. card_jg = [1, "合格"]
  796. # 经度
  797. lng_pass = [i for i in lng_lst if 113.76194444 >= i or i >= 113.77861111]
  798. if len(lng_pass):
  799. lng_jg = [4,f"{str(min(lng_lst))}~{str(max(lng_lst))},不符合条件的数据有{str(len(lng_pass))}条"]
  800. else:
  801. lng_jg = [1, f"{str(min(lng_lst))}~{str(max(lng_lst))}"]
  802. # 纬度 35.02083333,35.0375
  803. lat_pass = [i for i in lat_lst if 35.02083333 >=i or i >= 35.0375]
  804. if len(lat_pass):
  805. lat_jg = [4,f"{str(min(lat_lst))}~{str(max(lat_lst))},不符合条件的数据有{str(len(lat_pass))}条"]
  806. else:
  807. lat_jg = [1, f"{str(min(lat_lst))}~{str(max(lat_lst))}"]
  808. # 工作状态切换
  809. work_status_str = "".join(work_status)
  810. result = self.eliminate_adjacent_duplicates(work_status_str)
  811. count_01 = result.count("01")
  812. count_10 = result.count("10")
  813. ww = min([count_01, count_10])
  814. if ww >= 1:
  815. work_jg = [1, f"待机/工作转换{str(ww)}次"]
  816. else:
  817. work_jg = [0, f"无待机/工作转换"]
  818. # 电路板输入电压
  819. v_vs_pass = [i for i in v_vs_lst if 22 > i or i > 30]
  820. if len(v_vs_pass):
  821. v_vs = [0, f"{str(min(v_vs_lst))}~{str(max(v_vs_lst))},不合格数据3条"]
  822. else:
  823. v_vs = [1, f"{str(min(v_vs_lst))}~{str(max(v_vs_lst))}"]
  824. # 温控
  825. if "1" in tr_k_jg_lst:
  826. tr_k_jg = [1, f"上报被温控数据{str(tr_k_jg_lst.count('1'))}条"]
  827. else:
  828. tr_k_jg = [0, f"无数据上报"]
  829. # 雨控
  830. if self.set_product != "水稻测报灯":
  831. if "1" in rain_k_jg_lst:
  832. rain_k_jg = [1, f"被雨控有{str(rain_k_jg_lst.count('1'))}条数据"]
  833. else:
  834. rain_k_jg = [0, "无被雨控数据上报"]
  835. else:
  836. rain_k_jg = [1, "合格"]
  837. # 加热仓温度
  838. heat_temp = []
  839. for h in heat_temp_lst:
  840. if 85 <= h <= 105:
  841. heat_temp = [1, f"{str(min(heat_temp_lst))}~{str(max(heat_temp_lst))}"]
  842. break
  843. if not heat_temp:
  844. heat_temp = [0, f"{str(min(heat_temp_lst))}~{str(max(heat_temp_lst))}"]
  845. # 环境温度
  846. if {"at": "25", "ah": "35"} in temp_hum:
  847. en_temp = [0,"35且环境湿度显示25"]
  848. else:
  849. temp = [float(k.get("at", 0)) for k in temp_hum]
  850. en_temp = [1,f"{str(min(temp))}~{str(max(temp))}"]
  851. # 环境湿度
  852. if {"at": "35", "ah": "25"} in temp_hum:
  853. en_hum = [0,"25且环境温度显示35"]
  854. else:
  855. temp = [float(k.get("ah", 0)) for k in temp_hum]
  856. en_hum = [1,f"{str(min(temp))}~{str(max(temp))}"]
  857. # 图片质量
  858. if self.set_product == "4.0测报灯" or self.set_product == "水稻测报灯":
  859. sorted_dict = sorted(image_score.items(), key=lambda x: x[1], reverse=True)
  860. sorted_dict = sorted_dict[:3]
  861. lte = []
  862. score = []
  863. print(sorted_dict, "----------------")
  864. for v in sorted_dict:
  865. rank = 55 if self.set_product == "水稻测报灯" else 68
  866. if v[1] < rank:
  867. lte.append(str(v[1]))
  868. score.append(str(round(v[1], 3)))
  869. else:
  870. score.append(str(round(v[1], 3)))
  871. if sorted_dict:
  872. if lte:
  873. img_jg = [0, ",".join(score)]
  874. else:
  875. img_jg = [1, ",".join(score)]
  876. else:
  877. img_jg = [0, "输入开始时间前4h内无图片"]
  878. else:
  879. img_jg = [1, ""]
  880. # 上报信息条数
  881. interval = [abs(information_count[i+1] - information_count[i]) for i in range(len(information_count) - 1)]
  882. k = 0
  883. for i in interval:
  884. if i < 5 * 60:
  885. k += 1
  886. if len(information_count) >= 22 and k <= 3:
  887. infor_jg_count = [1, f"数据条数为{str(len(information_count))},其中有小于5分钟的数据间隔{str(k)}条"]
  888. elif len(information_count) >= 22 and k >=4:
  889. infor_jg_count = [0, f"数据条数{str(len(information_count))},其中有小于5分钟的数据间隔{str(k)}条"]
  890. else:
  891. infor_jg_count = [0, f"数据条数{str(len(information_count))},其中有小于5分钟的数据间隔{str(k)}条"]
  892. # 电路板固件版本号 RTU固件版本号
  893. rtus = []
  894. gujians = []
  895. for d in dver_lst:
  896. rtu = d.split("-")[-1]
  897. gujian = d.replace(rtu, "").replace("-", "")
  898. gujians.append(gujian)
  899. rtus.append(rtu)
  900. rtus = list(set(rtus))
  901. gujians = list(set(gujians))
  902. if len(gujians) == 1 and gujians[0] == self.set_stm8vs:
  903. gujian_num = [1, self.set_stm8vs]
  904. else:
  905. if len(gujians) == 1:
  906. gujian_num = [0, gujians[0]]
  907. else:
  908. gujian_num = [0, ",".join(gujians)]
  909. if len(rtus) == 1 and rtus[0] == self.set_dver:
  910. rtu_num = [1, self.set_dver]
  911. else:
  912. if len(rtus) == 1:
  913. rtu_num = [0, rtus[0]]
  914. else:
  915. rtu_num = [0, ",".join(rtus)]
  916. # 工作时长 判定
  917. if self.set_power == "DC":
  918. # total_time 去重
  919. total_time = list(set(total_time))
  920. if len(total_time) == 1 and total_time[0] == 4:
  921. # if tt == "4":
  922. time_jg = [1, "4"]
  923. else:
  924. time_jg = [0, "不合格"]
  925. else:
  926. time_jg = [0, "不合格"]
  927. # power_pass = []
  928. # for i in total_time:
  929. # if i not in power_pass:
  930. # power_pass.append(i)
  931. # if len(power_pass) == 1 and power_pass[0] == 4:
  932. # time_jg = [1, "4"]
  933. # else:
  934. # time_jg = [0, ",".join([str(i) for i in power_pass])]
  935. # if self.set_power == "AC":
  936. # power_pass = []
  937. # for i in total_time:
  938. # if i not in power_pass:
  939. # power_pass.append(i)
  940. # if len(power_pass) == 1 and power_pass[0] == 9:
  941. # time_jg = [1, "9"]
  942. # else:
  943. # time_jg = [0, ",".join([str(i) for i in power_pass])]
  944. if position == "河南省新乡市原阳县":
  945. position_cg = [1, position]
  946. else:
  947. position_cg = [0, position]
  948. return [
  949. device_id_info,
  950. [1, self.set_order],
  951. card_jg,
  952. lng_jg,
  953. lat_jg,
  954. work_jg,
  955. gujian_num,
  956. rtu_num,
  957. v_vs,
  958. tr_k_jg,
  959. rain_k_jg,
  960. infor_jg_count,
  961. heat_temp,
  962. en_temp,
  963. en_hum,
  964. img_jg,
  965. # current_jg,
  966. # up_jg,
  967. # down_jg,
  968. # lig_jg,
  969. # machine_jg,
  970. # camera_jg,
  971. time_jg,
  972. position_cg
  973. ]
  974. def __siqing_verify(self, data_result,status_result, status_all_result, alarm_result, device_id_info, image_score, process):
  975. """获取四情平台查询结果"""
  976. current_jg = [1,""] # 振动电机工作状态下电流匹配度
  977. up_jg = [1,"无电流不足信息"] # 上仓门电机电机工作状态
  978. down_jg = [1,"无电流不足信息"] # 下仓门电机电机工作状态
  979. lig_jg = [1,"无电流不足信息"] # 灯管工作状态下
  980. machine_jg = [1, "无电流不足信息"]# 电机工作状态
  981. camera_jg = [1, "无电流不足信息"] # 相机工作状态
  982. time_jg = [1, "合格"] # 工作时长
  983. current_jg_ct, up_jg_ct, down_jg_ct, lig_jg_ct, camera_jg_ct = 0,0,0,0,0
  984. if alarm_result:
  985. for alerm in alarm_result:
  986. a = alerm.get("alarm_desc", "")
  987. if a:
  988. try:
  989. a = eval(a)
  990. except Exception as e:
  991. a = json.loads(a)
  992. if a.get("type", "") == "震动电机":
  993. if a.get("status") == "电流不足":
  994. current_jg_ct += 1
  995. elif a.get("type", "") == "上仓门电机":
  996. if "电流不足" in a.get("status"):
  997. up_jg_ct += 1
  998. elif a.get("type", "") == "下仓门电机":
  999. if "电流不足" in a.get("status"):
  1000. down_jg_ct += 1
  1001. elif a.get("type", "") == "灯管":
  1002. if a.get("status") == "电流不足":
  1003. lig_jg_ct += 1
  1004. elif a.get("type", "") == "相机":
  1005. if a.get("status") == "电流不足":
  1006. camera_jg_ct += 1
  1007. if current_jg_ct:
  1008. current_jg = [1,""]
  1009. if up_jg_ct:
  1010. up_jg = [0,f"有电流不足信息{str(up_jg_ct)}条"]
  1011. if down_jg_ct:
  1012. down_jg = [0,f"有电流不足信息{str(down_jg_ct)}条"]
  1013. if lig_jg_ct:
  1014. lig_jg = [0,f"有电流不足信息{str(lig_jg_ct)}条"]
  1015. if camera_jg_ct:
  1016. camera_jg = [0,f"有电流不足信息{str(camera_jg_ct)}条"]
  1017. lng_lst = [] # 经度
  1018. lat_lst = [] # 纬度
  1019. tr_k_jg_lst = [] # 温控
  1020. rain_k_jg_lst = []# 雨控
  1021. infor_jg_count = 0 # 上报信息条数
  1022. heat_temp_lst = []# 加热仓温度
  1023. temp_hum = [] # 环境温度 环境湿度
  1024. total_time = [] # 总时长
  1025. information_count = [] # 上报时间
  1026. v_vs_lst = [] # 电压
  1027. for index, cur in enumerate(data_result):
  1028. upl_time_str = cur.get("upl_time", "0")
  1029. upl_time_int = int(time.mktime(upl_time_str.timetuple()))
  1030. information_count.append(upl_time_int)
  1031. device_data = cur.get("cbd_data", "")
  1032. if device_data:
  1033. device_data = ast.literal_eval(device_data)
  1034. lng = device_data.get("lng", "")
  1035. lng_lst.append(float(lng))
  1036. lat = device_data.get("lat", "")
  1037. lat_lst.append(float(lat))
  1038. v_vs_lst.append(float(device_data.get("vbat", "0")))
  1039. tr_k_jg_lst.append(device_data.get("tps", ""))
  1040. rain_k_jg_lst.append(device_data.get("rps", ""))
  1041. infor_jg_count += 1
  1042. heat_temp_lst.append(float(device_data.get("hrt", 0)))
  1043. temp_hum.append({"at": device_data.get("at", ""), "ah": device_data.get("ah", "")})
  1044. if self.set_power == "AC":
  1045. total_time.append(device_data.get("tt"))
  1046. elif self.set_power == "DC":
  1047. st = device_data.get("st", "00:00")
  1048. et = device_data.get("et", "00:00")
  1049. st = int(st.split(":")[0])
  1050. et = int(et.split(":")[0])
  1051. if st > et:
  1052. total = 24 - st + et
  1053. else:
  1054. total = et - st
  1055. total_time.append(total)
  1056. # 经度
  1057. lng_pass = [i for i in lng_lst if 113.76194444 >= i or i >= 113.77861111]
  1058. if len(lng_pass):
  1059. lng_jg = [0,f"{str(min(lng_lst))}~{str(max(lng_lst))},不符合条件的数据有{str(len(lng_pass))}条"]
  1060. else:
  1061. lng_jg = [1, f"{str(min(lng_lst))}~{str(max(lng_lst))}"]
  1062. # 纬度 35.02083333,35.0375
  1063. lat_pass = [i for i in lat_lst if 35.02083333 >=i or i >= 35.0375]
  1064. if len(lat_pass):
  1065. lat_jg = [0,f"{str(min(lat_lst))}~{str(max(lat_lst))},不符合条件的数据有{str(len(lat_pass))}条"]
  1066. else:
  1067. lat_jg = [1, f"{str(min(lat_lst))}~{str(max(lat_lst))}"]
  1068. # 电路板输入电压
  1069. v_vs_pass = [i for i in v_vs_lst if 22 >= i or i >= 30]
  1070. if len(v_vs_pass):
  1071. v_vs = [0, f"{str(min(v_vs_lst))}~{str(max(v_vs_lst))},不合格数据3条"]
  1072. else:
  1073. v_vs = [1, f"{str(min(v_vs_lst))}~{str(max(v_vs_lst))}"]
  1074. # 温控
  1075. if "1" in tr_k_jg_lst:
  1076. tr_k_jg = [1, f"上报被温控数据{str(tr_k_jg_lst.count('1'))}条"]
  1077. else:
  1078. tr_k_jg = [0, f"无数据上报"]
  1079. # 雨控
  1080. if self.set_product == "水稻测报灯":
  1081. rain_k_jg = [1, "合格"]
  1082. else:
  1083. if "1" in rain_k_jg_lst:
  1084. rain_k_jg = [1, f"被雨控有{str(rain_k_jg_lst.count('1'))}条数据"]
  1085. else:
  1086. rain_k_jg = [0, "无被雨控数据上报"]
  1087. # 加热仓温度
  1088. heat_temp = []
  1089. for h in heat_temp_lst:
  1090. if 85 <= h <= 105:
  1091. heat_temp = [1, f"{str(min(heat_temp_lst))}~{str(max(heat_temp_lst))}"]
  1092. break
  1093. if not heat_temp:
  1094. heat_temp = [0, f"{str(min(heat_temp_lst))}~{str(max(heat_temp_lst))}"]
  1095. # 环境温度
  1096. if {"at": "25", "ah": "35"} in temp_hum:
  1097. en_temp = [0,"35且环境湿度显示25"]
  1098. else:
  1099. temp = [float(k.get("at", 0)) for k in temp_hum]
  1100. en_temp = [1,f"{str(min(temp))}~{str(max(temp))}"]
  1101. # 环境湿度
  1102. if {"at": "35", "ah": "25"} in temp_hum:
  1103. en_hum = [0,"25且环境温度显示35"]
  1104. else:
  1105. temp = [float(k.get("ah", 0)) for k in temp_hum]
  1106. en_hum = [1,f"{str(min(temp))}~{str(max(temp))}"]
  1107. # 图片质量
  1108. img_pass = []
  1109. img_score = []
  1110. for k, v in image_score.items():
  1111. if v > 50:
  1112. img_score.append(v)
  1113. if v < 68:
  1114. img_pass.append((datetime.datetime.fromtimestamp(int(k))).strftime("%Y-%m-%d %H:%M:%S"))
  1115. if img_score:
  1116. if img_pass:
  1117. img_jg = [0, f"{str(min(img_score))}~{str(max(img_score))}, 不合格图片{str(len(img_pass))}幅, 上报时间为{','.join(img_pass)}"]
  1118. else:
  1119. img_jg = [1, f"{str(min(img_score))}~{str(max(img_score))}"]
  1120. else:
  1121. img_jg = [0, "图片得分均在50以下"]
  1122. # 上报信息条数
  1123. interval = [abs(information_count[i+1] - information_count[i]) for i in range(len(information_count) - 1)]
  1124. k = 0
  1125. for i in interval:
  1126. if i < 5 * 60:
  1127. k += 1
  1128. if len(information_count) >= 22 and k == 0:
  1129. infor_jg_count = [1, f"数据条数为{str(len(information_count))},无小于5分钟的数据间隔"]
  1130. elif len(information_count) >= 22 and k:
  1131. infor_jg_count = [0, f"数据条数{str(len(information_count))},其中有小于5分钟的数据间隔{str(k)}条"]
  1132. else:
  1133. infor_jg_count = [0, f"数据条数{str(len(information_count))}"]
  1134. iccid = ""
  1135. work_status = [] # 工作状态切换
  1136. elec_vs_lst = [] # 电路板固件版本号
  1137. for index, cur in enumerate(status_all_result):
  1138. device_data = cur.get("cbd_status", "")
  1139. if device_data:
  1140. device_data = ast.literal_eval(device_data)
  1141. iccid = device_data.get("iccid", "")
  1142. work_status.append(str(device_data.get("ws", "")))
  1143. elec_vs_lst.append(device_data.get("dver", ""))
  1144. # 检查卡号及有效期
  1145. card_jg = self.sim_info(iccid)
  1146. # 工作状态切换
  1147. work_status_str = "".join(work_status)
  1148. wr = work_status_str.count("01")
  1149. if wr > 4:
  1150. work_jg = [1, f"待机/工作转换{str(wr)}次"]
  1151. else:
  1152. work_jg = [0, f"待机/工作转换{str(wr)}次"]
  1153. # 电路板固件版本号 RTU固件版本号
  1154. rtus = []
  1155. gujians = []
  1156. for d in elec_vs_lst:
  1157. rtu = d.split("-")[-1]
  1158. gujian = d.replace(rtu, "").replace("-", "")
  1159. # if gujian == self.set_stm8vs:
  1160. gujians.append(gujian)
  1161. # if rtu == self.set_dver:
  1162. rtus.append(rtu)
  1163. rtus = list(set(rtus))
  1164. gujians = list(set(gujians))
  1165. if len(gujians) == 1 and gujians[0] == self.set_stm8vs:
  1166. gujian_num = [1, self.set_stm8vs]
  1167. else:
  1168. if len(gujians) == 1:
  1169. gujian_num = [0, gujians[0]]
  1170. else:
  1171. gujian_num = [0, ",".join(gujians)]
  1172. if len(rtus) == 1 and rtus[0] == self.set_dver:
  1173. rtu_num = [1, self.set_dver]
  1174. else:
  1175. if len(rtus) == 1:
  1176. rtu_num = [0, rtus[0]]
  1177. else:
  1178. rtu_num = [0, ",".join(rtus)]
  1179. # 工作时长 判定
  1180. if self.set_power == "DC":
  1181. power_pass = []
  1182. for i in total_time:
  1183. if i not in power_pass:
  1184. power_pass.append(i)
  1185. if len(power_pass) == 1 and power_pass[0] == 4:
  1186. time_jg = [1, "4"]
  1187. else:
  1188. time_jg = [0, ",".join([str(i) for i in power_pass])]
  1189. if self.set_power == "AC":
  1190. power_pass = []
  1191. for i in total_time:
  1192. if i not in power_pass:
  1193. power_pass.append(i)
  1194. if len(power_pass) == 1 and power_pass[0] == 9:
  1195. time_jg = [1, "9"]
  1196. else:
  1197. time_jg = [0, ",".join([str(i) for i in power_pass])]
  1198. return [
  1199. device_id_info,
  1200. [1, self.set_order],
  1201. card_jg,
  1202. lng_jg,
  1203. lat_jg,
  1204. work_jg,
  1205. gujian_num,
  1206. rtu_num,
  1207. v_vs,
  1208. tr_k_jg,
  1209. rain_k_jg,
  1210. infor_jg_count,
  1211. heat_temp,
  1212. en_temp,
  1213. en_hum,
  1214. current_jg,
  1215. up_jg,
  1216. down_jg,
  1217. lig_jg,
  1218. img_jg,
  1219. machine_jg,
  1220. camera_jg,
  1221. time_jg
  1222. ]
  1223. def run(self):
  1224. """主业务逻辑,涉及进度条不能模块化,慢慢捋"""
  1225. proess = 0
  1226. now_time = datetime.datetime.now()
  1227. time_stamp = str(int(datetime.datetime.timestamp(now_time)))
  1228. global save_filename
  1229. save_filename = self.set_order + "_" + now_time.strftime("%m%d") + f"测报灯检验-{time_stamp}.xlsx"
  1230. save_path = os.path.join(self.save_path,save_filename)
  1231. workbook = Workbook(save_path)
  1232. worksheet = workbook.add_worksheet()
  1233. worksheet.set_row(0, 30)
  1234. worksheet.set_row(1, 35)
  1235. worksheet.set_row(2, 50)
  1236. worksheet.set_row(3, 80)
  1237. for i in range(len(self.device_list)):
  1238. worksheet.set_row(i+4, 15)
  1239. worksheet.set_column(0, 26, 12)
  1240. toji_style = workbook.add_format(toji_format)
  1241. title_style = workbook.add_format(title_format)
  1242. explain_style = workbook.add_format(explain_formal)
  1243. formal_style = workbook.add_format(formal_format)
  1244. error_style = workbook.add_format(error_format)
  1245. common_style = workbook.add_format(common_format)
  1246. merge_title_style = workbook.add_format(merge_title_format)
  1247. worksheet.merge_range('A1:Y1',"物联网测报灯在线检验原始表格",merge_title_style)
  1248. # "振动电机工作状态下\n电流匹配度",
  1249. # "上仓门电机电机工作状态下 \n 电流匹配度","下仓门电机工作状态下 \n电流匹配度","灯管工作状态下 \n电流匹配度","清扫电机/转盘电机工作状态下的 \n 最大电流","相机工作状态下 \n电流匹配度",
  1250. title_data = [
  1251. "文档ID","设备ID", "任务单号", "卡号及有效期","经度","纬度","工作状态切换", "电路板固件版本号","RTU固件版本号",
  1252. "电路板输入电压","温控","雨控","上报信息条数","加热仓温度","环境温度","环境湿度","图片质量",
  1253. "工作时长", "位置", "综合判定"
  1254. ]
  1255. worksheet.write_row(row = 2 ,col = 0, data = title_data,cell_format=title_style)
  1256. explain_data = [
  1257. "/",
  1258. "后台显示与输入一致为合格,否则为不合格",
  1259. "按照输入文件的任务单为依据复制到此列",
  1260. "有显示卡号及有效期即为合格,否则为不合格",
  1261. "合格条件(绿色):\n113°46′13″±30″范围内,转为十进度为:(113.76194444,113.77861111)区间内",
  1262. "合格条件(绿色):\n35°1′45″±30″范围内,转为十进度为:(35.02083333,35.0375)区间内",
  1263. "合格条件(绿色):\n测试时间段内有待机/工作且出现大于等于4次以上循环",
  1264. "合格条件(绿色):\n后台上报结果与输入标准对照,相符则合格,其余则不合格",
  1265. "合格条件(绿色):\n后台上报结果与输入标准对照,相符则合格,其余则不合格",
  1266. "合格条件(绿色):\n电路板输入电压:22~30V",
  1267. "合格条件(绿色):\n测试时间段之前至少有一条被温控数据上报",
  1268. "合格条件(绿色):\n测试时间段内至少有一条被雨控数据上报",
  1269. "合格条件(绿色):\n1、“工作”的条数大于22条,2、在“工作”的前提下,两条数据间隔小于等于5分钟的次数不能高于3条,两项全部符合为合格,否则不合格",
  1270. "合格条件(绿色):\n上传的加热温度包含但不局限于85~105°之间的值",
  1271. "合格条件(绿色):\n等于35且环境温度等于25的前提下为不合格,其余显示的任何值为合格",
  1272. "合格条件(绿色):\n等于25度且环境湿度等于35的前提下为不合格,其余显示的任何值为合格",
  1273. "合格条件(绿色):\n对自动检查前上传的图片质量进行打分,取前三个图片分数,如果有一个小于68为不合格,否则合格",
  1274. # "合格条件(绿色):\n在工作时间段内没有接收到“电流不足”的信息为合格,否则不合格",
  1275. # "合格条件(绿色):\n在工作时间段内没有接收到“电流不足”的信息为合格,否则不合格",
  1276. # "合格条件(绿色):\n在工作时间段内没有接收到“电流不足”的信息为合格,否则不合格",
  1277. # "合格条件(绿色):\n在工作时间段内没有接收到“电流不足”的信息为合格,否则不合格",
  1278. # "合格条件(绿色):\n在工作时间段内没有接收到“电流不足”的信息为合格,否则不合格",
  1279. # "合格条件(绿色):\n在工作时间段内没有接收到“电流不足”的信息为合格,否则不合格",
  1280. "合格条件(绿色):\n与供电选择输入对照,(DC为4h,AC为9h)一致判定为合格,否则为不合格",
  1281. "位置信息(绿色): \n合格条件(绿色), 显示河南省新乡市原阳县为合格,否则不合格",
  1282. "合格条件(绿色):\nd:r列全部合格为合格,否则为不合格"
  1283. ]
  1284. worksheet.write_row(row = 3 ,col = 0, data = explain_data,cell_format=explain_style)
  1285. formal_counts = 0
  1286. verify_list = []
  1287. for index, short_id in enumerate(self.device_list):
  1288. short_id = str(int(short_id)).strip() # 把device_id转换成字符串并去除空格
  1289. d_id,deviceId,platform = self.device_their_platform(short_id)
  1290. # if short_id != deviceId:
  1291. # device_id_info = [0, "不合格"]
  1292. # else:
  1293. device_id_info = [1, deviceId]
  1294. worksheet.write(index+4,0,short_id,formal_style)
  1295. if platform == "大数据平台":
  1296. try:
  1297. worksheet.write(index+4,1,deviceId,formal_style)
  1298. self.mongo_ping()
  1299. data_status = self.cbd_alerm.find({"equip_id": deviceId, "alarm_time": {"$gte":self.start_time ,"$lte":self.end_time}}).sort([('alarm_time', pymongo.DESCENDING)])
  1300. data_cursor = self.cbd_collection.find({"device_id":d_id,'addtime': {"$gte":self.start_time ,"$lte":self.end_time}}).sort([('addtime', pymongo.DESCENDING)])
  1301. data_temps = self.cbd_collection.find({"device_id":d_id,'addtime': {"$gte":self.start_time - 3600 * 24 ,"$lte":self.start_time}}).sort([('addtime', pymongo.DESCENDING)])
  1302. data_counts = data_cursor.count()
  1303. # 配置
  1304. de = self.device_collection.find_one({"device_id":deviceId})
  1305. device_config = self.device_config.find_one({"d_id":de.get("id")})
  1306. tt = 0
  1307. if device_config:
  1308. device_config_temp = device_config.get("device_config")
  1309. device_config_temp = eval(device_config_temp)
  1310. device_config_temp_ext = device_config_temp.get("ext")
  1311. tt = device_config_temp_ext.get("tt")
  1312. image_score = self.__bigdata_iamge_verify(d_id)
  1313. if data_counts == 0:
  1314. for i in range(1,19):
  1315. worksheet.write(index+4,i,"搜索时间内无数据",error_style)
  1316. worksheet.write(index+4,19,"不合格",error_style)
  1317. danji_verify = []
  1318. for i in range(19):
  1319. danji_verify.append(0)
  1320. verify_list.append(danji_verify)
  1321. else:
  1322. danji_verify = []
  1323. result = self.get_position(deviceId)
  1324. verify_data = self.__bigdata_verify(data_cursor,data_status, device_id_info, image_score, proess, data_temps, result, tt)
  1325. for clo_index, verify_ in enumerate(verify_data):
  1326. danji_verify.append(verify_[0])
  1327. if verify_[0] == 0:
  1328. worksheet.write(index+4,clo_index+1,verify_[1],error_style)
  1329. elif verify_[0] == 1:
  1330. worksheet.write(index+4,clo_index+1,verify_[1],formal_style)
  1331. else:
  1332. worksheet.write(index+4,clo_index+1,verify_[1],common_style)
  1333. if 4 in danji_verify and (0 not in danji_verify):
  1334. worksheet.write(index+4,19,"可接受", formal_style)
  1335. elif 0 in danji_verify:
  1336. worksheet.write(index+4,19,"不合格", error_style)
  1337. else:
  1338. formal_counts += 1
  1339. worksheet.write(index+4,19,"合格",formal_style)
  1340. verify_list.append(danji_verify)
  1341. except Exception as e:
  1342. print(e)
  1343. elif platform == "四情平台":
  1344. try:
  1345. worksheet.write(index+4,1,deviceId,formal_style)
  1346. self.sql_ping()
  1347. data_sql = "SELECT * FROM AppInfoManage_cbddata WHERE equip_id_id='{}' AND upl_time BETWEEN '{}' AND '{}' order by upl_time;".format(deviceId,self.start_time_str,self.end_time_str)
  1348. alarm_sql = "SELECT * FROM AppInfoManage_cbd_alarm_record WHERE equip_id_id='{}' AND alarm_time BETWEEN '{}' AND '{}' order by alarm_time;".format(deviceId,self.start_time_str,self.end_time_str)
  1349. self.cursor.execute(data_sql)
  1350. data_result = self.cursor.fetchall()
  1351. self.cursor.execute(alarm_sql)
  1352. alarm_result = self.cursor.fetchall()
  1353. status_sql = "SELECT * FROM AppInfoManage_cbdstatus WHERE equip_id_id='{}' AND upl_time BETWEEN '{}' AND '{}' order by upl_time;".format(deviceId,self.start_time_str,self.end_time_str)
  1354. self.cursor.execute(status_sql)
  1355. status_result = self.cursor.fetchall()
  1356. status_all_sql = "SELECT * FROM AppInfoManage_cbdstatus_all WHERE equip_id_id='{}' AND upl_time BETWEEN '{}' AND '{}' order by upl_time;".format(deviceId,self.start_time_str,self.end_time_str)
  1357. self.cursor.execute(status_all_sql)
  1358. status_all_result = self.cursor.fetchall()
  1359. if len(data_result) == 0:
  1360. # 空白数据填充
  1361. for i in range(1,19):
  1362. worksheet.write(index+4,i,"搜索时间内无数据",error_style)
  1363. worksheet.write(index+4,19,"不合格",error_style)
  1364. danji_verify = []
  1365. for i in range(19):
  1366. danji_verify.append(0)
  1367. verify_list.append(danji_verify)
  1368. else:
  1369. image_score = self.__siqing_image_verify(deviceId)
  1370. verify_data = self.__siqing_verify(data_result,status_result, status_all_result, alarm_result, device_id_info, image_score, process)
  1371. danji_verify = []
  1372. for clo_index, verify_ in enumerate(verify_data):
  1373. danji_verify.append(verify_[0])
  1374. if verify_[0] == 0:
  1375. worksheet.write(index+4,clo_index+1,verify_[1],error_style)
  1376. elif verify_[0] == 1:
  1377. worksheet.write(index+4,clo_index+1,verify_[1],formal_style)
  1378. else:
  1379. worksheet.write(index+4,clo_index+1,verify_[1],common_style)
  1380. if 0 in danji_verify:
  1381. worksheet.write(index+4,19,"不合格",error_style)
  1382. else:
  1383. formal_counts += 1
  1384. worksheet.write(index+4,19,"合格",formal_style)
  1385. verify_list.append(danji_verify)
  1386. except Exception as e:
  1387. print(e)
  1388. pass
  1389. proess = (index+1)/len(self.device_list) * 100
  1390. if int(proess) == 100:
  1391. self.proess_signal.emit(99)
  1392. else:
  1393. self.proess_signal.emit(int(proess))
  1394. toji_data = [
  1395. "任务单号",self.set_order,
  1396. "检验时间",'{}\n{}'.format(self.start_time_str,self.end_time_str),
  1397. "检验数量",len(self.device_list),
  1398. "合格数",formal_counts,
  1399. "检验平台",self.set_plat,
  1400. "主板版本",self.set_stm8vs,
  1401. "联网版本",self.set_dver,
  1402. "产品名称", self.set_product,
  1403. "供电选择", self.set_power,
  1404. "报告日期",now_time.strftime("%y-%m-%d %H:%M:%S")
  1405. ]
  1406. worksheet.write_row(row = 1 ,col = 0, data = toji_data,cell_format=toji_style)
  1407. worksheet.protect(pwd_str) # 保护视图,不允许修改
  1408. workbook.close()
  1409. self.cursor.close()
  1410. self.connection.close()
  1411. self.myclient.close()
  1412. self.proess_signal.emit(100)
  1413. if __name__== "__main__":
  1414. QtWidgets.QApplication.setAttribute(QtCore.Qt.AA_EnableHighDpiScaling)
  1415. QtGui.QGuiApplication.setAttribute(QtCore.Qt.HighDpiScaleFactorRoundingPolicy.PassThrough)
  1416. app = QtWidgets.QApplication(sys.argv)
  1417. MainWindow = QtWidgets.QMainWindow()
  1418. ui = Ui_MainWindow()
  1419. ui.setupUi(MainWindow)
  1420. MainWindow.show()
  1421. sys.exit(app.exec_())