Spaces:
Running
Running
#!/usr/bin/env python | |
# -*- coding: utf-8 -*- | |
r""" | |
@DATE: 2024/9/5 19:32 | |
@File: face_detector.py | |
@IDE: pycharm | |
@Description: | |
人脸检测器 | |
""" | |
from mtcnnruntime import MTCNN | |
from .context import Context | |
from hivision.error import FaceError, APIError | |
from hivision.utils import resize_image_to_kb_base64 | |
import requests | |
import cv2 | |
import os | |
mtcnn = None | |
def detect_face_mtcnn(ctx: Context, scale: int = 2): | |
""" | |
基于MTCNN模型的人脸检测处理器,只进行人脸数量的检测 | |
:param ctx: 上下文,此时已获取到原始图和抠图结果,但是我们只需要原始图 | |
:param scale: 最大边长缩放比例,原图:缩放图 = 1:scale | |
:raise FaceError: 人脸检测错误,多个人脸或者没有人脸 | |
""" | |
global mtcnn | |
if mtcnn is None: | |
mtcnn = MTCNN() | |
image = cv2.resize( | |
ctx.origin_image, | |
(ctx.origin_image.shape[1] // scale, ctx.origin_image.shape[0] // scale), | |
interpolation=cv2.INTER_AREA, | |
) | |
faces, _ = mtcnn.detect(image, thresholds=[0.8, 0.8, 0.8]) | |
# print(len(faces)) | |
if len(faces) != 1: | |
# 保险措施,如果检测到多个人脸或者没有人脸,用原图再检测一次 | |
faces, _ = mtcnn.detect(ctx.origin_image) | |
else: | |
# 如果只有一个人脸,将人脸坐标放大 | |
for item, param in enumerate(faces[0]): | |
faces[0][item] = param * 2 | |
if len(faces) != 1: | |
raise FaceError("Expected 1 face, but got {}".format(len(faces)), len(faces)) | |
left = faces[0][0] | |
top = faces[0][1] | |
width = faces[0][2] - left + 1 | |
height = faces[0][3] - top + 1 | |
ctx.face = (left, top, width, height) | |
def detect_face_face_plusplus(ctx: Context): | |
""" | |
基于Face++ API接口的人脸检测处理器,只进行人脸数量的检测 | |
:param ctx: 上下文,此时已获取到原始图和抠图结果,但是我们只需要原始图 | |
:param scale: 最大边长缩放比例,原图:缩放图 = 1:scale | |
:raise FaceError: 人脸检测错误,多个人脸或者没有人脸 | |
""" | |
url = "https://api-cn.faceplusplus.com/facepp/v3/detect" | |
api_key = os.getenv("FACE_PLUS_API_KEY") | |
api_secret = os.getenv("FACE_PLUS_API_SECRET") | |
image = ctx.origin_image | |
# 将图片转为 base64, 且不大于2MB(Face++ API接口限制) | |
image_base64 = resize_image_to_kb_base64(image, 2000, mode="max") | |
files = { | |
"api_key": (None, api_key), | |
"api_secret": (None, api_secret), | |
"image_base64": (None, image_base64), | |
} | |
# 发送 POST 请求 | |
response = requests.post(url, files=files) | |
# 获取响应状态码 | |
status_code = response.status_code | |
response_json = response.json() | |
if status_code == 200: | |
face_num = response_json["face_num"] | |
if face_num == 1: | |
face_rectangle = response_json["faces"][0]["face_rectangle"] | |
ctx.face = ( | |
face_rectangle["left"], | |
face_rectangle["top"], | |
face_rectangle["width"], | |
face_rectangle["height"], | |
) | |
else: | |
raise FaceError( | |
"Expected 1 face, but got {}".format(face_num), len(face_num) | |
) | |
elif status_code == 401: | |
raise APIError( | |
f"Face++ Status code {status_code} Authentication error: API key and secret do not match.", | |
status_code, | |
) | |
elif status_code == 403: | |
reason = response_json.get("error_message", "Unknown authorization error.") | |
raise APIError( | |
f"Authorization error: {reason}", | |
status_code, | |
) | |
elif status_code == 400: | |
error_message = response_json.get("error_message", "Bad request.") | |
raise APIError( | |
f"Bad request error: {error_message}", | |
status_code, | |
) | |
elif status_code == 413: | |
raise APIError( | |
f"Face++ Status code {status_code} Request entity too large: The image exceeds the 2MB limit.", | |
status_code, | |
) | |