settings.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466
  1. from wsgiref import headers
  2. from dotenv import load_dotenv, find_dotenv
  3. from pathlib import Path # Python 3.6+ only
  4. import configparser, json, pytz
  5. import requests,os
  6. import pillow_avif
  7. from utils.common import message_queue
  8. import hashlib
  9. from pathlib import Path
  10. import os,stat
  11. TIME_ZONE = pytz.timezone("Asia/Shanghai")
  12. from numpy import true_divide
  13. from databases import (
  14. create_all_database,
  15. DeviceConfig,
  16. SysConfigs,
  17. CRUD,
  18. batch_insert_sys_configs,
  19. SqlQuery,
  20. batch_insert_device_configs,
  21. batch_insert_device_configsNew,
  22. )
  23. USER_TOKEN = None # 可以通过外部设置或从配置文件中加载
  24. USER_ENV = None # 可以通过外部设置或从配置文件中加载
  25. USER_CAMERAS = False # 可以通过外部设置或从配置文件中加载
  26. # 追加配置参数 machine_type 拍照机设备类型;0鞋;1服装
  27. MACHINE_TYPE = 0
  28. # 初始化数据表
  29. create_all_database()
  30. session = SqlQuery()
  31. device_config_crud = CRUD(DeviceConfig)
  32. all_devices = device_config_crud.read_all(session)
  33. if len(all_devices) == 0:
  34. # 如果配置表中一条数据都没有,就将初始化数据全部插入到数据表中
  35. action_tabs = json.load(open("action_tabs.json", encoding="utf-8"))
  36. actions = json.load(open("action.json", encoding="utf-8"))
  37. batch_insert_device_configs(session, action_tabs, actions)
  38. sys_config_crud = CRUD(SysConfigs)
  39. all_sys_configs = sys_config_crud.read_all(session)
  40. if len(all_sys_configs) == 0:
  41. # 如果配置表中一条数据都没有,就将初始化数据全部插入到数据表中
  42. sys_config_json = json.load(open("sys_configs.json", encoding="utf-8"))
  43. batch_insert_sys_configs(session, sys_config_json)
  44. else:
  45. # 查询数据库中缺失哪个数据,进行补充
  46. sys_config_json = json.load(open("sys_configs.json", encoding="utf-8"))
  47. for sys_config in sys_config_json:
  48. photos = CRUD(SysConfigs)
  49. item = photos.read(session, conditions={"key": sys_config.get("key")})
  50. if item == None:
  51. config = SysConfigs(**sys_config)
  52. session.add(config)
  53. session.commit() # 合并事务提交
  54. session.close()
  55. # 初始化数据表---结束
  56. def get_config_by_items(config_dict):
  57. __config_dict = {}
  58. for i, k in config_dict:
  59. __config_dict[i] = k
  60. return __config_dict
  61. keys = ["面料", "里料"]
  62. def sync_sys_configs2Online():
  63. hlm_token = USER_TOKEN
  64. headers = {
  65. "Authorization": f"Bearer {hlm_token}",
  66. "content-type": "application/json",
  67. }
  68. # 追加配置参数 machine_type 拍照机设备类型;0鞋;1服装
  69. session = SqlQuery()
  70. sysConfigs = CRUD(SysConfigs)
  71. all_configs = sysConfigs.read_all(session)
  72. localConfigData = {}
  73. for local_config in all_configs:
  74. localConfigData[local_config.key] = json.loads(local_config.value)
  75. data_json = json.dumps(
  76. {"configs": localConfigData, "machine_type": MACHINE_TYPE},
  77. ensure_ascii=False,
  78. )
  79. # 同步本地到线上
  80. url = DOMAIN + "/api/ai_image/camera_machine/update_all_user_configs"
  81. requests.post(url=url, headers=headers, data=data_json)
  82. session.close()
  83. def getSysConfigs(key, item, default=None):
  84. session = SqlQuery()
  85. crud = CRUD(SysConfigs)
  86. one_item = crud.read(session, conditions={"key": key})
  87. config = json.loads(one_item.value)
  88. session.close()
  89. if item == "image_out_format":
  90. default_format = config.get(item, default)
  91. if default_format == "" or default_format == None:
  92. return "png"
  93. return config.get(item, default)
  94. def updateSysConfigs(params):
  95. session = SqlQuery()
  96. sysConfig = CRUD(SysConfigs)
  97. # 如果value是字典或列表,需要转换为JSON字符串
  98. if "value" in params and isinstance(params["value"], (dict, list)):
  99. params["value"] = json.dumps(params["value"], ensure_ascii=False)
  100. # 走编辑逻辑
  101. # kws = params.__dict__
  102. sysConfig.updateConditions(session, conditions={"key": params["key"]}, **params)
  103. session.close()
  104. def get_dict_value(_dict, key, default=None):
  105. if key in _dict:
  106. return _dict[key]
  107. else:
  108. return default
  109. # 鞋设备为1 服装设备为5
  110. MCU_CODE = 1
  111. MACHINE_LEVEL = (
  112. "二档"
  113. if getSysConfigs("other_configs", "device_speed", "二档") == ""
  114. else getSysConfigs("other_configs", "device_speed", "二档")
  115. )
  116. IS_TEST = False
  117. IS_MCU = True
  118. IS_LIN_SHI_TEST = False
  119. PhotographSeconds = float(
  120. 0.5
  121. if getSysConfigs("take_photo_configs", "camera_delay", "0.5") == ""
  122. else getSysConfigs("take_photo_configs", "camera_delay", "0.5")
  123. ) # 拍照停留时间
  124. MAX_PIXIAN_SIZE = 12000000
  125. def moveSpeed(level: str = None):
  126. config = {
  127. "一档": {
  128. "camera_high_motor": {
  129. "max_speed": 10000,
  130. "up_speed": 800,
  131. "down_speed": 700,
  132. },
  133. "turntable_steering": {
  134. "max_speed": 6000,
  135. "up_speed": 500,
  136. "down_speed": 400,
  137. },
  138. },
  139. "二档": {
  140. "camera_high_motor": {
  141. "max_speed": 7000,
  142. "up_speed": 600,
  143. "down_speed": 500,
  144. },
  145. "turntable_steering": {
  146. "max_speed": 4500,
  147. "up_speed": 350,
  148. "down_speed": 300,
  149. },
  150. },
  151. "三档": {
  152. "camera_high_motor": {
  153. "max_speed": 3500,
  154. "up_speed": 400,
  155. "down_speed": 300,
  156. },
  157. "turntable_steering": {
  158. "max_speed": 3000,
  159. "up_speed": 200,
  160. "down_speed": 200,
  161. },
  162. },
  163. }
  164. if level is None:
  165. return config[MACHINE_LEVEL]
  166. else:
  167. return config[level]
  168. config = configparser.ConfigParser()
  169. config_name = "config.ini"
  170. config.read(config_name, encoding="utf-8")
  171. # 应用名称
  172. APP_NAME = config.get("app", "app_name")
  173. # 应用版本号
  174. APP_VERSION = config.get("app", "version")
  175. # 是否开启调试模式
  176. IS_DEBUG = config.get("app", "debug")
  177. IS_UPLOAD_HLM = True if config.get("app", "is_upload") == "true" else False
  178. # 应用端口号
  179. PORT = config.get("app", "port")
  180. # 应用线程数
  181. APP_WORKS = config.get("app", "works")
  182. # 应用host地址
  183. APP_HOST = config.get("app", "host")
  184. # 应用服务启动名称
  185. APP_RUN = config.get("app", "app_run")
  186. # 日志名称
  187. LOG_FILE_NAME = config.get("log", "log_file_name")
  188. # 最大字节数
  189. MAX_BYTES = config.get("log", "max_bytes")
  190. print("Max bytes is", MAX_BYTES)
  191. # 备份数量
  192. BACKUP_COUNTS = config.get("log", "backup_counts")
  193. # 远程服务器地址
  194. HLM_HOST = config.get("log", "hlm_host")
  195. PROJECT = config.get("app", "project")
  196. # ----------------------------------
  197. mcu_config_dict = config.items("mcu_config")
  198. _mcu_config_dict = {}
  199. for i, k in mcu_config_dict:
  200. _mcu_config_dict[i] = int(k)
  201. # print(_mcu_config_dict)
  202. _config_mcu_config = get_config_by_items(config.items("mcu_config"))
  203. LEFT_FOOT_ACTION = _mcu_config_dict["left_foot_action"]
  204. LEFT_FOOT_PHOTOGRAPH = _mcu_config_dict["left_foot_photograph"]
  205. LEFT_FOOT_ACTION_1 = _mcu_config_dict["left_foot_action_1"]
  206. LEFT_FOOT_ACTION_2 = _mcu_config_dict["left_foot_action_2"]
  207. RIGHT_FOOT_ACTION = _mcu_config_dict["right_foot_action"]
  208. RIGHT_FOOT_PHOTOGRAPH = _mcu_config_dict["right_foot_photograph"]
  209. RIGHT_FOOT_ACTION_1 = _mcu_config_dict["right_foot_action_1"]
  210. RIGHT_FOOT_ACTION_2 = _mcu_config_dict["right_foot_action_2"]
  211. NEXT_STEP = int(get_dict_value(_config_mcu_config, "next_step", 6)) # 下一步
  212. MOVE_UP = _mcu_config_dict["move_up"]
  213. MOVE_DOWN = _mcu_config_dict["move_down"]
  214. STOP = _mcu_config_dict["stop"]
  215. # camera_config_dict = config.items("camera_config")
  216. # _camera_config_dict = {}
  217. # for i, k in mcu_config_dict:
  218. # _camera_config_dict[i] = int(k)
  219. # _camera_config_dict = get_config_by_items(camera_config_dict)
  220. # LOW_ISO = _camera_config_dict["low_iso"]
  221. # HIGH_ISO = _camera_config_dict["high_iso"]
  222. DOMAIN = (
  223. "https://dev2.valimart.net"
  224. if config.get("app", "env") != "dev"
  225. else "https://dev2.pubdata.cn"
  226. )
  227. def getDoman(env):
  228. return "https://dev2.valimart.net" if env != "dev" else "https://dev2.pubdata.cn"
  229. Company = "惠利玛"
  230. is_test_plugins = true_divide
  231. OUT_PIC_MODE = "." + getSysConfigs("basic_configs", "image_out_format", "png") # ".png"
  232. OUT_PIC_SIZE = (
  233. [1600]
  234. if getSysConfigs("basic_configs", "main_image_size", [1600]) == ""
  235. else getSysConfigs("basic_configs", "main_image_size", [1600])
  236. ) # 主图大小
  237. Mode = getSysConfigs("other_configs", "product_type", "鞋类") # 程序执行类
  238. OUT_PIC_FACTOR = float(
  239. 1
  240. if getSysConfigs("basic_configs", "image_sharpening", "1") == ""
  241. else getSysConfigs("basic_configs", "image_sharpening", "1")
  242. ) # 图片锐化
  243. RESIZE_IMAGE_MODE = 1
  244. GRENERATE_MAIN_PIC_BRIGHTNESS = 254 # 色阶是否调整到位判断
  245. RUNNING_MODE = getSysConfigs("other_configs", "running_mode", "普通模式")
  246. DEFAULT_CUTOUT_MODE = getSysConfigs("other_configs", "cutout_mode", "普通抠图")
  247. CUTOUT_MODE = (
  248. 0 if getSysConfigs("other_configs", "cutout_mode", "普通抠图") == "普通抠图" else 1
  249. )
  250. import importlib
  251. OUT_PIC_QUALITY = "普通"
  252. GRENERATE_MAIN_PIC_BRIGHTNESS = int(
  253. getSysConfigs("other_configs", "grenerate_main_pic_brightness", 254)
  254. ) # 色阶是否调整到位判断
  255. SHADOW_PROCESSING = int(
  256. getSysConfigs("other_configs", "shadow_processing", 0)
  257. ) # 0表示要直线和曲线,1 表示只要直线
  258. LOWER_Y = int(
  259. getSysConfigs("other_configs", "lower_y", 4)
  260. ) # 鞋底以下多少距离作为阴影蒙版
  261. CHECK_LOWER_Y = int(
  262. getSysConfigs("other_configs", "check_lower_y", 4)
  263. ) # 检测亮度区域,倒数第几行
  264. IS_GET_GREEN_MASK = (
  265. True if getSysConfigs("other_configs", "is_get_green_mask", "否") == "是" else False
  266. ) # 是否进行绿幕抠图
  267. IMAGE_SAVE_MAX_WORKERS = int(
  268. getSysConfigs("other_configs", "image_save_max_workers", 20)
  269. ) # 批量保存的线程大小
  270. COLOR_GRADATION_CYCLES = int(
  271. getSysConfigs("other_configs", "color_gradation_cycles", 23)
  272. ) # 色阶处理循环次数
  273. # grenerate_main_pic_brightness 亮度范围 默认值254 范围 200 -255 步长 1
  274. # opacity 透明度 默认值 0.5 范围 0.5-1 步长 0.1
  275. # IMAGE_MASK_CONFIG = int(
  276. # getSysConfigs("basic_configs", "image_mask_config", {"mode":0,"opacity":0.5,"grenerate_main_pic_brightness":254})
  277. # ) # 色阶处理循环次数
  278. def recordDataPoint(token=None, page="", uuid=None, data=""):
  279. """记录日志"""
  280. if token == None:
  281. return
  282. if page == "":
  283. return
  284. if uuid == None:
  285. return
  286. headers = {"Content-Type": "application/json", "authorization": "Bearer " + token}
  287. params = {
  288. "site": 1,
  289. "type": 5,
  290. "channel": "aigc-camera",
  291. "uuid": uuid,
  292. "page": page,
  293. "page_url": "/",
  294. "describe": {"action": page, "data": data},
  295. "time": 0,
  296. }
  297. return requests.post(
  298. headers=headers, data=json.dumps(params), url=DOMAIN + "/api/record/point"
  299. )
  300. def syncPhotoRecord(data,action_type=0):
  301. """同步图片记录"""
  302. headers = {
  303. "Authorization": f"Bearer {USER_TOKEN}",
  304. "content-type": "application/json",
  305. }
  306. machine_type = MACHINE_TYPE
  307. url = getDoman(USER_ENV) + f"/api/ai_image/camera_machine/photo_records_handler"
  308. postData = {"data":data,"machine_type":machine_type,'action_type':action_type}
  309. response = requests.post(url=url,data=json.dumps(postData, ensure_ascii=False, default=str), headers=headers)
  310. print("用户token",response.content)
  311. def syncBatchPhotoRecordDelete():
  312. """同步图片记录"""
  313. headers = {
  314. "Authorization": f"Bearer {USER_TOKEN}",
  315. "content-type": "application/json",
  316. }
  317. machine_type = MACHINE_TYPE
  318. url = getDoman(USER_ENV) + f"/api/ai_image/camera_machine/photo_records_handler"
  319. postData = {"machine_type": machine_type, "action_type": 4}
  320. response = requests.post(
  321. url=url,
  322. data=json.dumps(postData, ensure_ascii=False, default=str),
  323. headers=headers,
  324. )
  325. print("用户token", response.content)
  326. def checkRecordSyncStatus():
  327. # check_photo_record_sync_status
  328. headers = {
  329. "Authorization": f"Bearer {USER_TOKEN}",
  330. "content-type": "application/json",
  331. }
  332. machine_type = MACHINE_TYPE
  333. url = getDoman(USER_ENV) + f"/api/ai_image/camera_machine/photo_records_handler?machine_type={machine_type}"
  334. try:
  335. response = requests.get(url=url, headers=headers).json()
  336. return response.get("data").get('status',True)
  337. except Exception as e:
  338. return True
  339. async def sendSocketMessage(
  340. code=0, msg="", data=None, device_status=2, msg_type=""
  341. ):
  342. data = {
  343. "code": code,
  344. "msg": msg,
  345. "status": device_status,
  346. "data": data,
  347. "msg_type": msg_type,
  348. }
  349. await message_queue.put(data)
  350. def calculate_md5(filepath):
  351. # 打开文件,以二进制只读模式打开
  352. with open(filepath, "rb") as f:
  353. # 创建MD5哈希对象
  354. md5hash = hashlib.md5()
  355. # 循环读取文件的内容并更新哈希对象
  356. for chunk in iter(lambda: f.read(4096), b""):
  357. md5hash.update(chunk)
  358. # 返回MD5哈希的十六进制表示
  359. return md5hash.hexdigest()
  360. __output_dir = config.get("output_config", "output_dir")
  361. path_obj = Path(os.path.abspath(__output_dir))
  362. OUTPUT_DIR = path_obj.absolute()
  363. print("OUTPUT_DIR",__output_dir,OUTPUT_DIR)
  364. def handle_remove_readonly(func, path, exc):
  365. os.chmod(path, stat.S_IWRITE)
  366. func(path)
  367. try:
  368. __scan_dir = config.get("scan_config", "scan_dir")
  369. if __scan_dir == "" or __scan_dir is None:
  370. SCAN_DIR = None
  371. else:
  372. scan_path_obj = Path(os.path.abspath(__scan_dir))
  373. SCAN_DIR = scan_path_obj.absolute()
  374. if not os.path.exists(SCAN_DIR):
  375. # 创建多级目录
  376. os.makedirs(SCAN_DIR, exist_ok=True)
  377. except:
  378. SCAN_DIR = None
  379. print('扫码配置不存在')
  380. print("SCAN_DIR",SCAN_DIR)
  381. CUSTOMER_TEMPLATE_URL = config.get("customer_template", "template_url")
  382. def hex_to_rgb(hex_color):
  383. """
  384. 将十六进制颜色值转换为RGB颜色值
  385. :param hex_color: 十六进制颜色值,例如 "#FF0000" 或 "FF0000"
  386. :return: RGB元组,例如 (255, 0, 0)
  387. """
  388. # 移除可能存在的 # 符号
  389. hex_color = hex_color.lstrip('#')
  390. # 确保是有效的十六进制字符串
  391. if len(hex_color) != 6:
  392. raise ValueError("十六进制颜色值格式不正确,应为6位十六进制数")
  393. # 将十六进制字符串转换为RGB值
  394. try:
  395. r = int(hex_color[0:2], 16)
  396. g = int(hex_color[2:4], 16)
  397. b = int(hex_color[4:6], 16)
  398. return (r, g, b)
  399. except ValueError:
  400. return (255, 255, 255)
  401. def rgb_to_hex(r, g, b):
  402. """
  403. 将RGB颜色值转换为十六进制颜色值
  404. :param r: 红色分量 (0-255)
  405. :param g: 绿色分量 (0-255)
  406. :param b: 蓝色分量 (0-255)
  407. :return: 十六进制颜色值,例如 "#FF0000"
  408. """
  409. # 验证RGB值在有效范围内
  410. if not all(0 <= val <= 255 for val in [r, g, b]):
  411. return "#FFFFFF"
  412. return "#{:02X}{:02X}{:02X}".format(r, g, b)