deal_image.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410
  1. import os,re
  2. from natsort import natsorted,ns
  3. import shutil
  4. import exifread
  5. import time
  6. import datetime
  7. from databases import PhotoRecord,SqlQuery,CRUD
  8. import settings
  9. import copy
  10. import json
  11. import requests
  12. from service.pic_deal import Picture
  13. import xlsxwriter
  14. from PIL import Image
  15. from .base_deal import BaseDealImage
  16. from middleware import UnicornException
  17. _Type = ['.png', '.PNG', '.jpg', '.JPG', '.gif', '.GIF', ".jpge", ".JPGE"]
  18. class DealImage(BaseDealImage):
  19. def __init__(self, image_dir=None):
  20. super().__init__()
  21. self.image_dir = image_dir
  22. self.header = None
  23. def check_path(self, image_dir: str):
  24. if not os.path.exists(image_dir):
  25. os.mkdir(image_dir)
  26. return True
  27. def list_dir(self, path):
  28. listdir = os.listdir(path)
  29. sort_result = sorted(listdir, key=lambda x: int(re.findall(r'^(\d+)_', x)[0]) if re.findall(r'^(\d+)_', x) else 0)
  30. print("listdir 排序排序排序排序排序排序", sort_result)
  31. # return natsorted(listdir, alg=ns.PATH)
  32. return sort_result
  33. def get_date_time_original(self, file_path):
  34. with open(file_path, 'rb') as file_data:
  35. tags = exifread.process_file(file_data)
  36. if "EXIF DateTimeOriginal" in tags:
  37. return str(tags["EXIF DateTimeOriginal"])
  38. else:
  39. return False
  40. def get_goods_art_no(self, goods_art_no):
  41. # time_array = time.strptime(date_time_original, "%Y:%m:%d %H:%M:%S")
  42. # time_array = time.mktime(time_array)
  43. # datetime_obj = datetime.datetime.fromtimestamp(time_array)
  44. session = SqlQuery()
  45. configModel = CRUD(PhotoRecord)
  46. result = configModel.read(
  47. session,
  48. conditions={"goods_art_no": goods_art_no},
  49. order_by="id",
  50. ascending=True,
  51. )
  52. session.close()
  53. if result:
  54. return result.goods_art_no, result.image_index, result.image_deal_mode
  55. else:
  56. return None
  57. def get_data_from_hqt_with_goods_art_no(self, goods_art_no_list):
  58. _goods_art_no_list = copy.deepcopy(goods_art_no_list)
  59. _list = []
  60. # 单次请求数少于20个
  61. goods_art_no_dict = {}
  62. while _goods_art_no_list:
  63. goods_art_no = _goods_art_no_list.pop()
  64. _list.append(goods_art_no)
  65. if len(_list) == 20 or len(_goods_art_no_list) == 0:
  66. online_goods_art_data = self.get_goods_art_no_info(
  67. goods_art_list=_list
  68. )
  69. if online_goods_art_data:
  70. for _goods_art_no in online_goods_art_data:
  71. goods_art_no_dict[_goods_art_no] = online_goods_art_data[
  72. _goods_art_no
  73. ]
  74. _list = []
  75. return goods_art_no_dict
  76. def get_goods_art_no_info(self, numbers_list=None, goods_art_list=None):
  77. # 获取商品基础信息,入参为商品的编号
  78. url = "{domain}/api/backend/goods_client/goods_query".format(
  79. domain=settings.DOMAIN
  80. )
  81. data = {
  82. 'goods_art_list': goods_art_list
  83. }
  84. data = json.dumps(data)
  85. _s = requests.session().post(url=url, data=data, headers=self.header)
  86. response_data = _s.json()
  87. goods_number_data = {}
  88. # ["", "", "", "", "", "", "", "", "", "", "", ]
  89. if "data" not in response_data:
  90. return {}
  91. for data in response_data["data"]:
  92. goods_number_data[data["goods_art_no"]] = {}
  93. goods_number_data[data["goods_art_no"]]["商品货号"] = data["goods_art_no"].upper()
  94. goods_number_data[data["goods_art_no"]]["款号"] = data["goods_number"].upper()
  95. goods_number_data[data["goods_art_no"]]["商品面料"] = data["fabric"]
  96. goods_number_data[data["goods_art_no"]]["商品内里"] = data["lining"]
  97. goods_number_data[data["goods_art_no"]]["商品鞋底"] = data["sole"]
  98. goods_number_data[data["goods_art_no"]]["鞋垫"] = data["insole"]
  99. goods_number_data[data["goods_art_no"]]["颜色名称"] = data["color"]
  100. return goods_number_data
  101. def get_data_from_hqt(self, goods_number_list):
  102. _goods_number_list = copy.deepcopy(goods_number_list)
  103. _list = []
  104. # 单次请求数少于20个
  105. goods_number_dict = {}
  106. while _goods_number_list:
  107. goods_art_no = _goods_number_list.pop()
  108. if "NUM" in goods_art_no:
  109. goods_art_no = goods_art_no.replace("NUM", "")
  110. _list.append(goods_art_no)
  111. if len(_list) == 20 or len(_goods_number_list) == 0:
  112. online_goods_art_data = self.get_goods_art_no_info(
  113. numbers_list=_list
  114. )
  115. if online_goods_art_data:
  116. for number in online_goods_art_data:
  117. goods_number_dict["NUM" + number] = online_goods_art_data[
  118. number
  119. ]
  120. _list = []
  121. return goods_number_dict
  122. def create_folder(self, path):
  123. def check_folder(__path):
  124. if not os.path.exists(__path):
  125. os.makedirs(__path)
  126. return False
  127. return True
  128. # 文件夹不存在,创建货号子集文件夹
  129. if not check_folder(path):
  130. for name in ["原始图", "原始图_已抠图", "800x800", "200images"]:
  131. other_path = path + "/" + name
  132. check_folder(other_path)
  133. def move_images(self, goods_art_no, goods_art_no_path, old_image_path):
  134. """
  135. 步骤:
  136. 1、移动到原始图
  137. Args:
  138. goods_art_no:
  139. goods_art_no_path:
  140. old_image_path:
  141. Returns:
  142. """
  143. # 移动到原始图
  144. file = os.path.split(old_image_path)[1]
  145. # 扩展名
  146. e = os.path.splitext(file)[1]
  147. print("self.goods_images_count_dict", self.goods_images_count_dict)
  148. # 获取图片序列
  149. self.goods_images_count_dict[goods_art_no] += 1
  150. # A9999(1).jpg
  151. new_file_name = "{}({})".format(goods_art_no, self.goods_images_count_dict[goods_art_no])
  152. original_image_path = "{}/原始图/{}{}".format(goods_art_no_path, new_file_name, e)
  153. try:
  154. shutil.move(old_image_path, original_image_path)
  155. except Exception as e:
  156. print(f"文件操作异常:{e}")
  157. def dealMoveImage(self, image_dir: str, callback_func=None,goods_art_no=None) -> dict:
  158. if not self.check_path(image_dir=image_dir + "/历史"):
  159. raise UnicornException("文件夹创建失败")
  160. # 遍历目标文件夹,获取有拍摄信息的图片,并按拍摄时间排序
  161. files = self.list_dir(image_dir)
  162. original_photo_list = [] # 原始图片列表
  163. for file_idx,file in enumerate(files):
  164. # -----图片清洗
  165. file_path = image_dir + "/" + file
  166. if os.path.isdir(file_path): # 忽略文件夹
  167. continue
  168. file_name, suffix = os.path.splitext(file)
  169. if suffix not in _Type: # 非图片进行移除
  170. shutil.move(file_path, image_dir + "/历史/" + file)
  171. continue
  172. date_time_original = self.get_date_time_original(file_path) # 获取照片拍照时间
  173. if date_time_original:
  174. # 基于照片的时间,与数据库匹配goods_art_no
  175. _data = self.get_goods_art_no(goods_art_no)
  176. if _data:
  177. # 能匹配上数据库
  178. goods_art_no, image_index, image_deal_mode = _data
  179. print("832 与数据库匹配goods_art_no", file_path,file_name, date_time_original, goods_art_no)
  180. original_photo_list.append({"file_path": file_path,
  181. "file": file,
  182. "date_time_original": date_time_original,
  183. "goods_art_no": goods_art_no,
  184. "image_index": file_idx,
  185. "real_goods_art_no": "",
  186. "real_goods_number": "",
  187. })
  188. else:
  189. # 匹配不上报错
  190. # self.show_progress_detail("图片:{} 无法对应货号,不做处理".format(file))
  191. if callback_func:
  192. callback_func("图片:{} 无对应货号".format(file))
  193. # shutil.move(photo_dict["file_path"], self.image_dir + "/历史/" + photo_dict["file"])
  194. continue
  195. else:
  196. shutil.move(file_path, image_dir + "/历史/" + file)
  197. if not original_photo_list:
  198. raise UnicornException("没有任何匹配的图片")
  199. return False, "没有任何匹配的图片"
  200. # 排序需要基于拍照的文件序号进行处理
  201. # original_photo_list.sort(
  202. # key=lambda x: "{}-{}-{}".format(x["goods_art_no"], x["image_index"], x["file"]))
  203. # print(original_photo_list)
  204. # 对有拍摄信息的图片进行数据库比对,如有比对上,则移动至货号文件夹,否则移入历史文件夹
  205. total_num = len(original_photo_list)
  206. # 当天日期作为文件夹
  207. seconds = time.time()
  208. output_path = "output/{f_name}".format(f_name=time.strftime("%Y-%m-%d", time.localtime(seconds)))
  209. # 遍历每个匹配好的数据进行处理
  210. print(f"original_photo_list:",original_photo_list)
  211. n = 0
  212. for photo_dict in original_photo_list:
  213. n += 1
  214. print("处理dict:",photo_dict)
  215. # 进度条
  216. goods_art_no = photo_dict["goods_art_no"]
  217. original_image_path = photo_dict["file_path"]
  218. # 输出货号文件夹
  219. if photo_dict["real_goods_art_no"]:
  220. goods_art_no = "{}@{}".format(photo_dict["real_goods_art_no"], photo_dict["real_goods_number"])
  221. goods_art_no_path = "{output_path}/{goods_art_no}".format(output_path=output_path,
  222. goods_art_no=goods_art_no)
  223. # 创建货号下的一系列文件夹
  224. self.create_folder(goods_art_no_path)
  225. # 重命名并进行移动
  226. print("开始移动:{} {} 命名为:{}".format(goods_art_no, original_image_path, goods_art_no_path))
  227. self.move_images(goods_art_no, goods_art_no_path, original_image_path) # 货号、货号文件路径、原始图路径
  228. time.sleep(0.2)
  229. # self.progress_sign.emit({"type": "移动原始图片", "progress_bar_value": int(n / total_num * 100)})
  230. # self.show_progress_detail("货号{} 相关文件夹创建完成,已移动原图~".format(goods_art_no))
  231. if callback_func:
  232. callback_func("货号{} 相关文件夹创建完成,已移动原图~".format(goods_art_no))
  233. print("已完成移动处理")
  234. if n != 0:
  235. # if settings.MattingPics:
  236. # # 检查所有未处理的货号文件夹,查看是否有完成图片加工处理
  237. # self.deal_images()
  238. # 自动生成一个货号表
  239. print("output_path", output_path)
  240. self.deal(output_path)
  241. # 完成处理
  242. # self.set_state(state_value=2)
  243. return True, output_path
  244. def deal(cls, dir_path):
  245. # print("dir_path", dir_path)
  246. out_excel_data = []
  247. for goods_art_no_folder in os.listdir(dir_path): # 遍历货号文件夹集合
  248. if not os.path.isdir(
  249. "{}/{}".format(dir_path, goods_art_no_folder)
  250. ): # 非文件夹进行过滤
  251. continue
  252. if "软件" in goods_art_no_folder:
  253. continue
  254. # print("goods_art_no_folder", goods_art_no_folder)
  255. # 如果存在800的主图,则优先进行使用
  256. big_image_folder_path = "{}/{}/800x800".format(
  257. dir_path, goods_art_no_folder
  258. )
  259. if not os.path.exists(big_image_folder_path):
  260. os.mkdir(big_image_folder_path)
  261. all_big_images = os.listdir(big_image_folder_path)
  262. goods_pic_total = len(all_big_images)
  263. _Type = [".png", ".PNG", ".jpg", ".JPG", ".gif", ".GIF", ".jpge", ".JPGE"]
  264. thumb_image_file_path = None
  265. # print("all_big_images",all_big_images)
  266. if all_big_images:
  267. for _file in all_big_images:
  268. # print(_file)
  269. file_name, e = os.path.splitext(_file)
  270. # print(file_name, e)
  271. if e in _Type:
  272. thumb_image_file_path = "{}/{}/800x800/{}".format(
  273. dir_path, goods_art_no_folder, _file
  274. )
  275. break
  276. # 如果不存在主图则进行使用原始图
  277. if thumb_image_file_path is None:
  278. _path = "{}/{}/原始图".format(dir_path, goods_art_no_folder)
  279. if not os.path.exists(_path):
  280. continue
  281. all_original_images = os.listdir(_path) # 遍历货号原始图文件夹
  282. goods_pic_total = len(all_original_images)
  283. if not all_original_images:
  284. continue
  285. image_file = all_original_images[0] # 取第一个货号图
  286. image_file_path = "{}/{}/原始图/{}".format(
  287. dir_path, goods_art_no_folder, image_file
  288. )
  289. if not os.path.exists(
  290. "{}/{}/200images".format(dir_path, goods_art_no_folder)
  291. ):
  292. os.mkdir("{}/{}/200images".format(dir_path, goods_art_no_folder))
  293. thumb_image_file_path = "{}/{}/200images/{}".format(
  294. dir_path, goods_art_no_folder, image_file
  295. )
  296. if not os.path.exists(thumb_image_file_path):
  297. # 开始触发进行压缩生成文件
  298. shutil.copy(image_file_path, thumb_image_file_path) # 复制文件
  299. pic = Picture(thumb_image_file_path)
  300. pic.resize(width=600)
  301. pic.save_img(thumb_image_file_path)
  302. # print("thumb_image_file_path", thumb_image_file_path)
  303. goods_number = ""
  304. if "@" in goods_art_no_folder:
  305. _ = goods_art_no_folder.split("@")
  306. goods_art_no_folder = _[0]
  307. goods_number = _[1].replace("NUM", "")
  308. out_excel_data.append(
  309. [
  310. goods_number,
  311. goods_art_no_folder,
  312. thumb_image_file_path,
  313. goods_pic_total,
  314. ]
  315. )
  316. if out_excel_data:
  317. out_excel_path = "{}/货号表-{}.xlsx".format(dir_path, time.time())
  318. options = {
  319. "default_format_properties": {
  320. "align": "left",
  321. "valign": "vcenter",
  322. "text_wrap": True,
  323. }
  324. }
  325. book = xlsxwriter.Workbook(filename=out_excel_path, options=options)
  326. sheet = book.add_worksheet("sheet1")
  327. # sheet.freeze_panes(1, 2)
  328. sheet.set_column("B:B", 17)
  329. sheet.set_column("D:D", 20)
  330. sheet.write_row("A1", ["编号", "原货号", "新货号", "缩略图", "原始图张数"])
  331. for index, data in enumerate(out_excel_data):
  332. # print(data)
  333. goods_number, goods_art_no, image_file, goods_pic_total = data
  334. try:
  335. im = Image.open(image_file)
  336. im_x, im_y = im.size
  337. image_width = 100
  338. image_height = int(im_y * image_width / im_x)
  339. sheet.set_row(index + 1, 95)
  340. x_scale = round(
  341. image_width / im_x, 2
  342. ) # 固定宽度/要插入的原始图片宽
  343. y_scale = round(
  344. image_height / im_y, 2
  345. ) # 固定高度/要插入的原始图片高
  346. sheet.insert_image(
  347. index + 1,
  348. 3,
  349. image_file,
  350. {
  351. "x_scale": x_scale,
  352. "y_scale": y_scale,
  353. "x_offset": 5,
  354. "y_offset": 5,
  355. "object_position":1,
  356. },
  357. )
  358. except:
  359. pass
  360. sheet.write_row("A{}".format(index + 2), [goods_number])
  361. sheet.write_row("B{}".format(index + 2), [goods_art_no])
  362. sheet.write_row("E{}".format(index + 2), [goods_pic_total])