From eb15a09f1f6d108342a59663919afab941710b69 Mon Sep 17 00:00:00 2001 From: joizhang Date: Thu, 18 Nov 2021 10:36:30 +0800 Subject: [PATCH 1/6] feat: face recognition pytorch version. --- README.md | 8 ++++---- app/__init__.py | 2 +- app/recognition/{face_net.py => facenet_tf.py} | 0 app/recognition/facenet_torch.py | 0 tests/test_env.py | 8 ++++++-- tests/{test_face_net.py => test_facenet_tf.py} | 2 +- 6 files changed, 12 insertions(+), 8 deletions(-) rename app/recognition/{face_net.py => facenet_tf.py} (100%) create mode 100644 app/recognition/facenet_torch.py rename tests/{test_face_net.py => test_facenet_tf.py} (96%) diff --git a/README.md b/README.md index bb9cd06..2edbf8d 100644 --- a/README.md +++ b/README.md @@ -32,7 +32,7 @@ Face recognition server end #### 参与贡献 -1. Fork 本仓库 -2. 添加修改代码 -3. 提交代码 -4. 新建 Pull Request \ No newline at end of file +1. Fork 本仓库 +2. 添加修改代码 +3. 提交代码 +4. 新建 Pull Request diff --git a/app/__init__.py b/app/__init__.py index fd327b9..902d4b5 100644 --- a/app/__init__.py +++ b/app/__init__.py @@ -8,7 +8,7 @@ from flask_moment import Moment from flask_sqlalchemy import SQLAlchemy from config import app_config -from app.recognition.face_net import FaceNet +from app.recognition.facenet_tf import FaceNet db = SQLAlchemy() moment = Moment() diff --git a/app/recognition/face_net.py b/app/recognition/facenet_tf.py similarity index 100% rename from app/recognition/face_net.py rename to app/recognition/facenet_tf.py diff --git a/app/recognition/facenet_torch.py b/app/recognition/facenet_torch.py new file mode 100644 index 0000000..e69de29 diff --git a/tests/test_env.py b/tests/test_env.py index 0fe967a..4e27364 100644 --- a/tests/test_env.py +++ b/tests/test_env.py @@ -1,14 +1,18 @@ import unittest import tensorflow.compat.v1 as tf +import torch -class TFEnvTestCase(unittest.TestCase): +class EnvTestCase(unittest.TestCase): - def test_env(self): + def test_tensorflow_env(self): gpu_options = tf.GPUOptions(per_process_gpu_memory_fraction=0.5) with tf.Session(config=tf.ConfigProto(gpu_options=gpu_options)) as sess: a = tf.constant([1.0, 2.0, 3.0, 4.0, 5.0, 6.0], shape=[2, 3], name='a') b = tf.constant([1.0, 2.0, 3.0, 4.0, 5.0, 6.0], shape=[3, 2], name='b') c = tf.matmul(a, b) self.assertEqual([[22., 28.], [49., 64.]], sess.run(c).tolist()) + + def test_pytorch_env(self): + self.assertTrue(torch.cuda.is_available()) diff --git a/tests/test_face_net.py b/tests/test_facenet_tf.py similarity index 96% rename from tests/test_face_net.py rename to tests/test_facenet_tf.py index f847a46..a4f282e 100644 --- a/tests/test_face_net.py +++ b/tests/test_facenet_tf.py @@ -4,7 +4,7 @@ from pathlib import Path import numpy as np import tensorflow.compat.v1 as tf -from app.recognition.face_net import FaceNet +from app.recognition.facenet_tf import FaceNet from app.utils import file_utils FILE_PREFIX = Path(Path(__file__).resolve().drive + '/face-recognition-data') -- Gitee From 0f7c5e0f73e0d7b40859ab14505440c4ced6609d Mon Sep 17 00:00:00 2001 From: joizhang Date: Thu, 18 Nov 2021 15:05:34 +0800 Subject: [PATCH 2/6] fix: tensorflow unresolved warning. --- app/recognition/facenet_tf.py | 3 ++- tests/test_facenet_tf.py | 3 ++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/app/recognition/facenet_tf.py b/app/recognition/facenet_tf.py index ef4dc35..25efa06 100644 --- a/app/recognition/facenet_tf.py +++ b/app/recognition/facenet_tf.py @@ -6,6 +6,7 @@ import logging import numpy as np import tensorflow.compat.v1 as tf +from tensorflow.compat.v1 import gfile LOG = logging.getLogger(__name__) @@ -55,7 +56,7 @@ class FaceNet: def create_inference(self, model_path): # await asyncio.sleep(1) - with tf.gfile.FastGFile(model_path, "rb") as f: + with gfile.FastGFile(model_path, "rb") as f: graph_def = tf.GraphDef() graph_def.ParseFromString(f.read()) tf.import_graph_def(graph_def) diff --git a/tests/test_facenet_tf.py b/tests/test_facenet_tf.py index a4f282e..f997aac 100644 --- a/tests/test_facenet_tf.py +++ b/tests/test_facenet_tf.py @@ -3,6 +3,7 @@ from pathlib import Path import numpy as np import tensorflow.compat.v1 as tf +from tensorflow.compat.v1 import gfile from app.recognition.facenet_tf import FaceNet from app.utils import file_utils @@ -16,7 +17,7 @@ class FaceNetTestCase(unittest.TestCase): def test_model_node(self): model_path = str(MODEL_PATH_PREFIX / '20180402-114759' / '20180402-114759.pb') - with tf.gfile.FastGFile(model_path, "rb") as f: + with gfile.FastGFile(model_path, "rb") as f: graph_def = tf.GraphDef() graph_def.ParseFromString(f.read()) tf.import_graph_def(graph_def) -- Gitee From 2e9419a4c30655d2155b8be2d4979929f7874eae Mon Sep 17 00:00:00 2001 From: joizhang Date: Thu, 18 Nov 2021 21:05:10 +0800 Subject: [PATCH 3/6] feat: Add pytorch facenet. --- app/__init__.py | 8 ++- app/api/__init__.py | 1 + app/api/face.py | 39 +++++++++++++ app/api/users.py | 32 +--------- app/recognition/face_recognition.py | 11 ++++ app/recognition/{facenet_tf.py => facenet.py} | 57 +++++++++++++----- app/recognition/facenet_torch.py | 0 app/utils/file_utils.py | 58 +++++++++++-------- config.py | 1 + tests/{test_facenet_tf.py => test_facenet.py} | 15 ++++- 10 files changed, 150 insertions(+), 72 deletions(-) create mode 100644 app/api/face.py create mode 100644 app/recognition/face_recognition.py rename app/recognition/{facenet_tf.py => facenet.py} (61%) delete mode 100644 app/recognition/facenet_torch.py rename tests/{test_facenet_tf.py => test_facenet.py} (75%) diff --git a/app/__init__.py b/app/__init__.py index 902d4b5..fe665ac 100644 --- a/app/__init__.py +++ b/app/__init__.py @@ -8,11 +8,12 @@ from flask_moment import Moment from flask_sqlalchemy import SQLAlchemy from config import app_config -from app.recognition.facenet_tf import FaceNet +from app.recognition.facenet import FaceNetTF, FaceNetTorch db = SQLAlchemy() moment = Moment() -face_net = FaceNet() +face_net_tf = FaceNetTF() +face_net_torch = FaceNetTorch() def create_app(config): @@ -29,7 +30,8 @@ def create_app(config): db.init_app(app) moment.init_app(app) if not app.testing: - face_net.init_app(app) + face_net_tf.init_app(app) + face_net_torch.init_app(app) from .errors import bp as errors_bp app.register_blueprint(errors_bp) diff --git a/app/api/__init__.py b/app/api/__init__.py index 90286e6..8466333 100644 --- a/app/api/__init__.py +++ b/app/api/__init__.py @@ -4,3 +4,4 @@ bp = Blueprint('api', __name__) from . import users from . import foo +from . import face diff --git a/app/api/face.py b/app/api/face.py new file mode 100644 index 0000000..74e860f --- /dev/null +++ b/app/api/face.py @@ -0,0 +1,39 @@ +import logging +import pickle + +import numpy as np +from flask import jsonify, request +from werkzeug.exceptions import BadRequest + +from app import face_net_tf +from app.model.models import User, Photo +from app.recognition import compare +from app.utils import file_utils +from . import bp + +LOG = logging.getLogger(__name__) + + +@bp.route('/face/recognition', methods=['POST']) +def face_recognition(): + data = request.json or {} + if 'photo' not in data: + raise BadRequest('Must include photo.') + photo = data['photo'] + input_images = [file_utils.resize_blob_to_160x160(photo)] + embedding = face_net_tf.get_embeddings(input_images) + photos = Photo.query.all() + x, y = [], [] + for p in photos: + x.append(pickle.loads(p.embedding)) + y.append(p.user_id) + x = np.array(x) + y = np.array(y) + LOG.info([embedding.shape, x.shape, y.shape]) + target = compare.face_net_compare(embedding, x, y) + if target == 0: + user = User() + user.from_dict({'id': 0, 'name': '无法识别', 'cell_phone_number': ''}) + else: + user = User.query.filter_by(id=int(target)).first() + return jsonify(success=True, data=user.to_dict(), status_code=200) diff --git a/app/api/users.py b/app/api/users.py index e7f367b..bc1756c 100644 --- a/app/api/users.py +++ b/app/api/users.py @@ -1,15 +1,12 @@ import logging -import pickle -import numpy as np from flask import jsonify, request from werkzeug.exceptions import BadRequest -from app import db, face_net +from app import db, face_net_tf from app.model.models import User, Photo from app.utils import file_utils, crypto_utils from . import bp -from app.recognition import compare LOG = logging.getLogger(__name__) @@ -40,35 +37,10 @@ def add_user_face(): photo_count = Photo.query.filter_by(id=user.id).count() for i in range(len(photos)): input_images = [file_utils.resize_blob_to_160x160(photos[i])] - embedding = face_net.get_embeddings(input_images)[0].dumps() + embedding = face_net_tf.get_embeddings(input_images)[0].dumps() photo_name = crypto_utils.encrypt(user.name + '-' + user.cell_phone_number + '-' + str(i + 1 + photo_count)) file_utils.write_webcam_blob(photos[i], photo_name) p = Photo(name=photo_name, embedding=embedding, user_id=user.id) db.session.add(p) db.session.commit() return jsonify(success=True, data=None, status_code=200) - - -@bp.route('/user/recognition', methods=['POST']) -def face_recognition(): - data = request.json or {} - if 'photo' not in data: - raise BadRequest('Must include photo.') - photo = data['photo'] - input_images = [file_utils.resize_blob_to_160x160(photo)] - embedding = face_net.get_embeddings(input_images) - photos = Photo.query.all() - x, y = [], [] - for p in photos: - x.append(pickle.loads(p.embedding)) - y.append(p.user_id) - x = np.array(x) - y = np.array(y) - LOG.info([embedding.shape, x.shape, y.shape]) - target = compare.face_net_compare(embedding, x, y) - if target == 0: - user = User() - user.from_dict({'id': 0, 'name': '无法识别', 'cell_phone_number': ''}) - else: - user = User.query.filter_by(id=int(target)).first() - return jsonify(success=True, data=user.to_dict(), status_code=200) diff --git a/app/recognition/face_recognition.py b/app/recognition/face_recognition.py new file mode 100644 index 0000000..1c5d46f --- /dev/null +++ b/app/recognition/face_recognition.py @@ -0,0 +1,11 @@ +class FaceRecognition: + __shared_state = {} + + def __init__(self): + self.__dict__ = self.__shared_state + + def init_app(self, app): + raise NotImplementedError + + def get_embeddings(self, input_images): + raise NotImplementedError diff --git a/app/recognition/facenet_tf.py b/app/recognition/facenet.py similarity index 61% rename from app/recognition/facenet_tf.py rename to app/recognition/facenet.py index 25efa06..4d7699e 100644 --- a/app/recognition/facenet_tf.py +++ b/app/recognition/facenet.py @@ -4,13 +4,19 @@ from __future__ import print_function import logging +import numpy import numpy as np import tensorflow.compat.v1 as tf +import torch.cuda +from facenet_pytorch import InceptionResnetV1 from tensorflow.compat.v1 import gfile +from torchvision.transforms import transforms + +from .face_recognition import FaceRecognition LOG = logging.getLogger(__name__) -CONFIG_OPTIONS = ['FACE_NET_MODEL_PATH'] +CONFIG_OPTIONS = ['FACE_NET_MODEL_PATH', 'FACE_NET_TORCH_HOME'] DEFAULT_OPTIONS = dict(model_path='') @@ -30,23 +36,17 @@ def get_app_dict(app_instance): } -class FaceNet: - __shared_state = {} - - def __init__(self, app=None, phase_train_placeholder=False, images_placeholder=None, embeddings=None, sess=None): - self.__dict__ = self.__shared_state - self.state = 'Init' - self.phase_train_placeholder = phase_train_placeholder - self.images_placeholder = images_placeholder - self.embeddings = embeddings - self.sess = sess +class FaceNetTF(FaceRecognition): + def __init__(self, app=None): + super().__init__() + self.phase_train_placeholder = False + self.images_placeholder = None + self.embeddings = None + self.sess = None if app is not None: self.init_app(app) - def __str__(self): - return self.state - def init_app(self, app): if 'FACE_NET_MODEL_PATH' not in app.config: LOG.error('Please config model path') @@ -78,3 +78,32 @@ class FaceNet: for embedding in embeddings: str_embeddings.append(str(embedding, encoding='utf8')) return str_embeddings + + +class FaceNetTorch(FaceRecognition): + + def __init__(self, app=None): + super().__init__() + self.model = None + self.transform = transforms.Compose([ + transforms.ToTensor(), + transforms.Normalize(mean=[0.485, 0.456, 0.406], std=[0.229, 0.224, 0.225]) + ]) + if app is not None: + self.init_app(app) + + def init_app(self, app): + self.create_model() + + def create_model(self): + # set TORCH_HOME in your os environment + self.model = InceptionResnetV1(pretrained='vggface2') + assert torch.cuda.is_available() + self.model.cuda() + self.model.eval() + + def get_embeddings(self, input_images: numpy.ndarray): + x = self.transform(input_images) + x = x.unsqueeze(0).cuda() + embeddings = self.model(x) + return embeddings diff --git a/app/recognition/facenet_torch.py b/app/recognition/facenet_torch.py deleted file mode 100644 index e69de29..0000000 diff --git a/app/utils/file_utils.py b/app/utils/file_utils.py index 7edb689..6813968 100644 --- a/app/utils/file_utils.py +++ b/app/utils/file_utils.py @@ -2,7 +2,7 @@ import base64 import re from pathlib import Path -import cv2 +import cv2.cv2 as cv2 import numpy as np FILE_PREFIX = Path(Path(__file__).resolve().drive + '/face-recognition-data') @@ -22,12 +22,12 @@ def write_webcam_blob(blob: str, file_name: str) -> bool: data = result.groupdict().get("data") else: raise Exception("Do not parse!") - img = base64.urlsafe_b64decode(data) + image = base64.urlsafe_b64decode(data) if not FILE_PREFIX.exists(): FILE_PREFIX.mkdir(parents=True) write_file_path = "{}.{}".format(str(FILE_PREFIX / file_name), ext) with open(write_file_path, 'wb') as f: - f.write(img) + f.write(image) return True @@ -41,33 +41,45 @@ def image_to_base64(image_path) -> str: return image_base64_format -def image_decode(base64_data: str): - img = base64.b64decode(base64_data.split(',')[1]) - np_data = np.fromstring(img, np.uint8) +def decode_image(base64_data: str): + image = str(base64.b64decode(base64_data.split(',')[1])) + np_data = np.fromstring(image, np.uint8) return cv2.imdecode(np_data, cv2.COLOR_BGR2RGB) -def _resize_to_160x160(img): - height, width = img.shape[:2] - res = img - if height < FACE_NET_INPUT_SIZE or width < FACE_NET_INPUT_SIZE: - diff = FACE_NET_INPUT_SIZE - min(height, width) - res = cv2.resize(img, (width + diff, height + diff), interpolation=cv2.INTER_AREA) - elif height == width: - res = cv2.resize(img, (FACE_NET_INPUT_SIZE, FACE_NET_INPUT_SIZE), interpolation=cv2.INTER_AREA) - height, width = res.shape[:2] +def _isotropically_resize_image(image, size, interpolation_down=cv2.INTER_AREA, interpolation_up=cv2.INTER_CUBIC): + h, w = image.shape[:2] + if max(w, h) == size: + return image + if w > h: + scale = size / w + h = h * scale + w = size + else: + scale = size / h + w = w * scale + h = size + interpolation = interpolation_up if scale > 1 else interpolation_down + resized = cv2.resize(image, (int(w), int(h)), interpolation=interpolation) + return resized + + +def _resize_to_160x160(image: np.ndarray) -> np.ndarray: + image = _isotropically_resize_image(image, FACE_NET_INPUT_SIZE) + height, width = image.shape[:2] x_center, y_center = height // 2, width // 2 x1, y1, x2, y2 = x_center - 80, y_center - 80, x_center + 80, y_center + 80 assert x2 - x1 == 160 and y2 - y1 == 160 - res = res[x1:x2, y1:y2] - return res + cropped = image[x1:x2, y1:y2] + return cropped -def resize_blob_to_160x160(blob: str): - img = image_decode(blob) - return _resize_to_160x160(img) +def resize_blob_to_160x160(blob: str) -> np.ndarray: + image = decode_image(blob) + return _resize_to_160x160(image) -def resize_image_to_160x160(image: str): - img = cv2.imread(image) - return _resize_to_160x160(img) +def resize_image_to_160x160(image_path: str) -> np.ndarray: + image = cv2.imread(image_path, cv2.IMREAD_COLOR) + image = cv2.cvtColor(image, cv2.COLOR_BGR2RGB) + return _resize_to_160x160(image) diff --git a/config.py b/config.py index 30c76bc..6fc7e5e 100644 --- a/config.py +++ b/config.py @@ -15,6 +15,7 @@ class Config: SQLALCHEMY_ECHO = (os.getenv("SQLALCHEMY_ECHO") == 'true') SQLALCHEMY_TRACK_MODIFICATIONS = (os.getenv("SQLALCHEMY_TRACK_MODIFICATIONS") == 'true') FACE_NET_MODEL_PATH = os.getenv('FACE_NET_MODEL_PATH') + FACE_NET_TORCH_HOME = os.getenv('TORCH_HOME') @staticmethod def init_app(app): diff --git a/tests/test_facenet_tf.py b/tests/test_facenet.py similarity index 75% rename from tests/test_facenet_tf.py rename to tests/test_facenet.py index f997aac..c33efae 100644 --- a/tests/test_facenet_tf.py +++ b/tests/test_facenet.py @@ -5,7 +5,7 @@ import numpy as np import tensorflow.compat.v1 as tf from tensorflow.compat.v1 import gfile -from app.recognition.facenet_tf import FaceNet +from app.recognition.facenet import FaceNetTF, FaceNetTorch from app.utils import file_utils FILE_PREFIX = Path(Path(__file__).resolve().drive + '/face-recognition-data') @@ -36,8 +36,19 @@ class FaceNetTestCase(unittest.TestCase): image = str(FILE_PREFIX / 'test1-15922865715-1.jpg') res = file_utils.resize_image_to_160x160(image) model_path = str(MODEL_PATH_PREFIX / '20170512-110547' / '20170512-110547.pb') - face_net = FaceNet() + face_net = FaceNetTF() face_net.create_inference(model_path) embedding = face_net.get_embeddings(np.array([res]))[0] self.assertEqual(128, len(embedding)) print(embedding.dumps()) + + +class FaceNetTorchTestCase(unittest.TestCase): + + def test_get_embedding(self): + image_path = str(FILE_PREFIX / 'Aaron_Eckhart_0001.jpg') + image = file_utils.resize_image_to_160x160(image_path) + face_net = FaceNetTorch() + face_net.create_model() + embedding = face_net.get_embeddings(image) + self.assertEqual(512, len(embedding[0])) -- Gitee From cbf02cfbb23eaf5ef7ce1df7e39b99b073570871 Mon Sep 17 00:00:00 2001 From: joizhang Date: Fri, 19 Nov 2021 10:24:15 +0800 Subject: [PATCH 4/6] feat: Add date of creation. --- app/api/users.py | 16 ++++++++++++++-- app/model/models.py | 4 +++- sql/scheme.sql | 39 ++++----------------------------------- 3 files changed, 21 insertions(+), 38 deletions(-) diff --git a/app/api/users.py b/app/api/users.py index bc1756c..d80687d 100644 --- a/app/api/users.py +++ b/app/api/users.py @@ -19,8 +19,8 @@ def get_users(): return jsonify(success=True, data=data, status_code=200) -@bp.route('/user/face', methods=['POST']) -def add_user_face(): +@bp.route('/user', methods=['POST']) +def add_user(): data = request.json or {} if 'name' not in data or 'cell_phone_number' not in data or 'photos' not in data: raise BadRequest('Must include name, cell_phone_number and photos fields.') @@ -44,3 +44,15 @@ def add_user_face(): db.session.add(p) db.session.commit() return jsonify(success=True, data=None, status_code=200) + + +@bp.route('/user', methods=['DELETE']) +def delete_user(): + user_id = request.args.get('id', default=0, type=int) + if user_id == 0: + raise BadRequest('Invalid Request!') + u = User() + u.id = user_id + db.session.delete(u) + db.session.commit() + return jsonify(success=True, data=None, status_code=200) diff --git a/app/model/models.py b/app/model/models.py index a33e891..2c85a6a 100644 --- a/app/model/models.py +++ b/app/model/models.py @@ -1,6 +1,7 @@ from flask import url_for from app import db +from datetime import datetime class PaginatedAPIMixin(object): @@ -32,6 +33,7 @@ class User(PaginatedAPIMixin, db.Model): id = db.Column(db.Integer, primary_key=True) name = db.Column(db.String(32), nullable=False) cell_phone_number = db.Column(db.String(32), nullable=False) + create_date = db.Columb(db.DateTime, default=datetime.utcnow) photos = db.relationship('Photo', backref='user', lazy=True) def __repr__(self): @@ -54,9 +56,9 @@ class User(PaginatedAPIMixin, db.Model): class Photo(db.Model): __tablename__ = 't_photo' id = db.Column(db.Integer, primary_key=True) + user_id = db.Column(db.Integer, nullable=False) name = db.Column(db.String(512), nullable=False) embedding = db.Column(db.BLOB(), nullable=False) - user_id = db.Column(db.Integer, db.ForeignKey('t_user.id'), nullable=False) def __repr__(self): return ''.format(self.photo_path) diff --git a/sql/scheme.sql b/sql/scheme.sql index 2cdaea6..46d1f72 100644 --- a/sql/scheme.sql +++ b/sql/scheme.sql @@ -1,56 +1,25 @@ -- 用户表 +drop table if exists t_user; create table t_user ( id int not null auto_increment, name varchar(32) not null, cell_phone_number varchar(32) not null, + create_date datetime not null, primary key (`id`) ) ENGINE = InnoDB AUTO_INCREMENT = 1 DEFAULT CHARSET = utf8mb4; --- 签到表 t_sign_in -# create table t_sign_in -# ( -# id int not null primary key auto_increment, -# name int not null, -# start_time datetime not null, -# end_time datetime not null -# ); - --- 用户照片表 -# drop table if exists t_user_photo; -# create table t_user_photo -# ( -# user_id int not null, -# photo_id int not null, -# constraint userid2 foreign key (user_id) references t_user (id), -# constraint photoid2 foreign key (photo_id) references t_photo (id) -# ); - - -- 照片表 drop table if exists t_photo; create table t_photo ( id int not null auto_increment, + user_id int not null, name varchar(512) not null, embedding blob not null, - user_id int not null, - primary key (`id`), - constraint userid foreign key (user_id) references t_user (id) + primary key (`id`) ) ENGINE = InnoDB AUTO_INCREMENT = 1 DEFAULT CHARSET = utf8mb4; - --- 用户签到表 t_user_sign -# create table t_user_sign -# ( -# user_id int not null, -# sign_id int not null, -# constraint userid3 foreign key (user_id) references t_user (id), -# constraint signid3 foreign key (sign_id) references t_sign_in (id) -# ); - - - -- Gitee From da634ca6113dffc7a02f90d9813d4e59154da073 Mon Sep 17 00:00:00 2001 From: joizhang Date: Fri, 19 Nov 2021 10:55:04 +0800 Subject: [PATCH 5/6] feat: Apply facenet pytorch. --- app/__init__.py | 4 ++-- app/api/users.py | 14 +++++++------- app/model/models.py | 2 +- app/recognition/face_recognition.py | 5 ++++- app/recognition/facenet.py | 21 ++++++++------------- tests/test_facenet.py | 18 +++++++++--------- 6 files changed, 31 insertions(+), 33 deletions(-) diff --git a/app/__init__.py b/app/__init__.py index fe665ac..6be8d26 100644 --- a/app/__init__.py +++ b/app/__init__.py @@ -12,7 +12,7 @@ from app.recognition.facenet import FaceNetTF, FaceNetTorch db = SQLAlchemy() moment = Moment() -face_net_tf = FaceNetTF() +# face_net_tf = FaceNetTF() face_net_torch = FaceNetTorch() @@ -30,7 +30,7 @@ def create_app(config): db.init_app(app) moment.init_app(app) if not app.testing: - face_net_tf.init_app(app) + # face_net_tf.init_app(app) face_net_torch.init_app(app) from .errors import bp as errors_bp diff --git a/app/api/users.py b/app/api/users.py index d80687d..1721818 100644 --- a/app/api/users.py +++ b/app/api/users.py @@ -3,7 +3,7 @@ import logging from flask import jsonify, request from werkzeug.exceptions import BadRequest -from app import db, face_net_tf +from app import db, face_net_torch from app.model.models import User, Photo from app.utils import file_utils, crypto_utils from . import bp @@ -30,17 +30,17 @@ def add_user(): u = User() u.from_dict(data) db.session.add(u) - db.session.commit() user = u # Photo photos = data['photos'] photo_count = Photo.query.filter_by(id=user.id).count() for i in range(len(photos)): - input_images = [file_utils.resize_blob_to_160x160(photos[i])] - embedding = face_net_tf.get_embeddings(input_images)[0].dumps() - photo_name = crypto_utils.encrypt(user.name + '-' + user.cell_phone_number + '-' + str(i + 1 + photo_count)) + input_image = file_utils.resize_blob_to_160x160(photos[i]) + embedding = face_net_torch.get_embeddings(input_image) + photo_name = f'{user.name}-{user.cell_phone_number}-{str(i + 1 + photo_count)}' + photo_name = crypto_utils.encrypt(photo_name) file_utils.write_webcam_blob(photos[i], photo_name) - p = Photo(name=photo_name, embedding=embedding, user_id=user.id) + p = Photo(name=photo_name, embedding=embedding.dumps(), user_id=user.id) db.session.add(p) db.session.commit() return jsonify(success=True, data=None, status_code=200) @@ -50,7 +50,7 @@ def add_user(): def delete_user(): user_id = request.args.get('id', default=0, type=int) if user_id == 0: - raise BadRequest('Invalid Request!') + raise BadRequest('Illegal Request!') u = User() u.id = user_id db.session.delete(u) diff --git a/app/model/models.py b/app/model/models.py index 2c85a6a..67cc7dc 100644 --- a/app/model/models.py +++ b/app/model/models.py @@ -33,7 +33,7 @@ class User(PaginatedAPIMixin, db.Model): id = db.Column(db.Integer, primary_key=True) name = db.Column(db.String(32), nullable=False) cell_phone_number = db.Column(db.String(32), nullable=False) - create_date = db.Columb(db.DateTime, default=datetime.utcnow) + create_date = db.Column(db.DateTime, default=datetime.utcnow) photos = db.relationship('Photo', backref='user', lazy=True) def __repr__(self): diff --git a/app/recognition/face_recognition.py b/app/recognition/face_recognition.py index 1c5d46f..3d804b4 100644 --- a/app/recognition/face_recognition.py +++ b/app/recognition/face_recognition.py @@ -1,3 +1,6 @@ +import numpy + + class FaceRecognition: __shared_state = {} @@ -7,5 +10,5 @@ class FaceRecognition: def init_app(self, app): raise NotImplementedError - def get_embeddings(self, input_images): + def get_embeddings(self, input_image) -> numpy.ndarray: raise NotImplementedError diff --git a/app/recognition/facenet.py b/app/recognition/facenet.py index 4d7699e..44dc824 100644 --- a/app/recognition/facenet.py +++ b/app/recognition/facenet.py @@ -67,17 +67,11 @@ class FaceNetTF(FaceRecognition): self.sess = tf.Session() self.sess.run(tf.global_variables_initializer()) - def get_embeddings(self, input_images): - feed_dict = {self.images_placeholder: np.array(input_images), self.phase_train_placeholder: False} + def get_embeddings(self, input_image: np.ndarray): + x = np.expand_dims(input_image, axis=0) + feed_dict = {self.images_placeholder: x, self.phase_train_placeholder: False} embeddings = self.sess.run(self.embeddings, feed_dict=feed_dict) - return embeddings - - def get_str_embeddings(self, input_images): - embeddings = self.get_embeddings(input_images) - str_embeddings = [] - for embedding in embeddings: - str_embeddings.append(str(embedding, encoding='utf8')) - return str_embeddings + return embeddings[0] class FaceNetTorch(FaceRecognition): @@ -102,8 +96,9 @@ class FaceNetTorch(FaceRecognition): self.model.cuda() self.model.eval() - def get_embeddings(self, input_images: numpy.ndarray): - x = self.transform(input_images) + def get_embeddings(self, input_image: numpy.ndarray): + x = self.transform(input_image) x = x.unsqueeze(0).cuda() embeddings = self.model(x) - return embeddings + embeddings = embeddings.detach().cpu().numpy() + return embeddings[0] diff --git a/tests/test_facenet.py b/tests/test_facenet.py index c33efae..040a10b 100644 --- a/tests/test_facenet.py +++ b/tests/test_facenet.py @@ -15,8 +15,13 @@ MODEL_PATH_PREFIX = Path(__file__).parents[1] class FaceNetTestCase(unittest.TestCase): + def test_model_exists(self): + model_path = MODEL_PATH_PREFIX / '20170512-110547' / '20170512-110547.pb' + print(model_path) + self.assertTrue(model_path.exists()) + def test_model_node(self): - model_path = str(MODEL_PATH_PREFIX / '20180402-114759' / '20180402-114759.pb') + model_path = str(MODEL_PATH_PREFIX / '20170512-110547' / '20170512-110547.pb') with gfile.FastGFile(model_path, "rb") as f: graph_def = tf.GraphDef() graph_def.ParseFromString(f.read()) @@ -27,18 +32,13 @@ class FaceNetTestCase(unittest.TestCase): for operation in operations: print(operation.name) - def test_model_exists(self): - model_path = MODEL_PATH_PREFIX / '20180402-114759' / '20180402-114759.pb' - print(model_path) - self.assertTrue(model_path.exists()) - def test_get_embeddings(self): - image = str(FILE_PREFIX / 'test1-15922865715-1.jpg') - res = file_utils.resize_image_to_160x160(image) + image_path = str(FILE_PREFIX / 'Aaron_Eckhart_0001.jpg') + image = file_utils.resize_image_to_160x160(image_path) model_path = str(MODEL_PATH_PREFIX / '20170512-110547' / '20170512-110547.pb') face_net = FaceNetTF() face_net.create_inference(model_path) - embedding = face_net.get_embeddings(np.array([res]))[0] + embedding = face_net.get_embeddings(np.array([image]))[0] self.assertEqual(128, len(embedding)) print(embedding.dumps()) -- Gitee From 97f284b17191bb1c19da17e19554f6581ca971f1 Mon Sep 17 00:00:00 2001 From: joizhang Date: Fri, 19 Nov 2021 15:22:55 +0800 Subject: [PATCH 6/6] feat: Delete user. --- app/api/face.py | 6 ++--- app/api/users.py | 19 +++++++++------ app/model/models.py | 1 - app/utils/file_utils.py | 51 +++++++++++++++++++++++++--------------- tests/test_file_utils.py | 10 ++++---- 5 files changed, 52 insertions(+), 35 deletions(-) diff --git a/app/api/face.py b/app/api/face.py index 74e860f..f17a572 100644 --- a/app/api/face.py +++ b/app/api/face.py @@ -5,7 +5,7 @@ import numpy as np from flask import jsonify, request from werkzeug.exceptions import BadRequest -from app import face_net_tf +from app import face_net_torch from app.model.models import User, Photo from app.recognition import compare from app.utils import file_utils @@ -20,8 +20,8 @@ def face_recognition(): if 'photo' not in data: raise BadRequest('Must include photo.') photo = data['photo'] - input_images = [file_utils.resize_blob_to_160x160(photo)] - embedding = face_net_tf.get_embeddings(input_images) + input_image = file_utils.resize_blob_to_160x160(photo) + embedding = face_net_torch.get_embeddings(input_image) photos = Photo.query.all() x, y = [], [] for p in photos: diff --git a/app/api/users.py b/app/api/users.py index 1721818..c107ddd 100644 --- a/app/api/users.py +++ b/app/api/users.py @@ -30,16 +30,17 @@ def add_user(): u = User() u.from_dict(data) db.session.add(u) - user = u + user = User.query.filter_by(name=data['name'], cell_phone_number=data['cell_phone_number']).first() # Photo photos = data['photos'] photo_count = Photo.query.filter_by(id=user.id).count() for i in range(len(photos)): input_image = file_utils.resize_blob_to_160x160(photos[i]) embedding = face_net_torch.get_embeddings(input_image) - photo_name = f'{user.name}-{user.cell_phone_number}-{str(i + 1 + photo_count)}' - photo_name = crypto_utils.encrypt(photo_name) - file_utils.write_webcam_blob(photos[i], photo_name) + data, ext = file_utils.split_data_ext(photos[i]) + photo_name = '{}-{}-{}'.format(user.name, user.cell_phone_number, str(i + 1 + photo_count)) + photo_name = '{}.{}'.format(crypto_utils.encrypt(photo_name), ext) + file_utils.write_photo(data, photo_name) p = Photo(name=photo_name, embedding=embedding.dumps(), user_id=user.id) db.session.add(p) db.session.commit() @@ -51,8 +52,12 @@ def delete_user(): user_id = request.args.get('id', default=0, type=int) if user_id == 0: raise BadRequest('Illegal Request!') - u = User() - u.id = user_id - db.session.delete(u) + photos = Photo.query.filter_by(user_id=user_id).all() + if photos is not None: + for photo in photos: + file_utils.remove_photo(photo.name) + db.session.delete(photo) + user = User.query.filter_by(id=user_id).first() + db.session.delete(user) db.session.commit() return jsonify(success=True, data=None, status_code=200) diff --git a/app/model/models.py b/app/model/models.py index 67cc7dc..e324546 100644 --- a/app/model/models.py +++ b/app/model/models.py @@ -34,7 +34,6 @@ class User(PaginatedAPIMixin, db.Model): name = db.Column(db.String(32), nullable=False) cell_phone_number = db.Column(db.String(32), nullable=False) create_date = db.Column(db.DateTime, default=datetime.utcnow) - photos = db.relationship('Photo', backref='user', lazy=True) def __repr__(self): return ''.format(self.name) diff --git a/app/utils/file_utils.py b/app/utils/file_utils.py index 6813968..c3ba57a 100644 --- a/app/utils/file_utils.py +++ b/app/utils/file_utils.py @@ -1,4 +1,5 @@ import base64 +import os.path import re from pathlib import Path @@ -10,27 +11,38 @@ FILE_PREFIX = Path(Path(__file__).resolve().drive + '/face-recognition-data') FACE_NET_INPUT_SIZE = 160 -def write_webcam_blob(blob: str, file_name: str) -> bool: - """Store file to disk, the name is "name-cell_phone_number-index.webp" - :param blob: webp data - :param file_name: username - :return: - """ +def split_data_ext(blob): result = re.search("data:image/(?P.*?);base64,(?P.*)", blob, re.DOTALL) if result: ext = result.groupdict().get("ext") data = result.groupdict().get("data") else: - raise Exception("Do not parse!") + raise ValueError("Do not parse!") + return data, ext + + +def write_photo(data: str, file_name: str) -> bool: + """Store file to disk, the name is "name-cell phone number-index.webp" + :param data: webp data + :param file_name: username + :return: + """ image = base64.urlsafe_b64decode(data) if not FILE_PREFIX.exists(): FILE_PREFIX.mkdir(parents=True) - write_file_path = "{}.{}".format(str(FILE_PREFIX / file_name), ext) - with open(write_file_path, 'wb') as f: + file_path = FILE_PREFIX / file_name + with open(file_path, 'wb') as f: f.write(image) return True +def remove_photo(file_name: str) -> bool: + file_path = FILE_PREFIX / file_name + if file_path.exists(): + os.remove(file_path) + return True + + def image_to_base64(image_path) -> str: """Transfer imagee to base64 string""" ext = image_path.split('.')[-1] @@ -42,23 +54,24 @@ def image_to_base64(image_path) -> str: def decode_image(base64_data: str): - image = str(base64.b64decode(base64_data.split(',')[1])) - np_data = np.fromstring(image, np.uint8) - return cv2.imdecode(np_data, cv2.COLOR_BGR2RGB) + image = base64.b64decode(base64_data.split(',')[1]) + image = np.frombuffer(image, np.uint8) + return cv2.imdecode(image, cv2.IMREAD_COLOR) def _isotropically_resize_image(image, size, interpolation_down=cv2.INTER_AREA, interpolation_up=cv2.INTER_CUBIC): + """Ensure min side >= size""" h, w = image.shape[:2] - if max(w, h) == size: + if min(w, h) == size: return image - if w > h: - scale = size / w - h = h * scale - w = size - else: + if h < w: scale = size / h w = w * scale h = size + else: + scale = size / w + h = h * scale + w = size interpolation = interpolation_up if scale > 1 else interpolation_down resized = cv2.resize(image, (int(w), int(h)), interpolation=interpolation) return resized @@ -82,4 +95,4 @@ def resize_blob_to_160x160(blob: str) -> np.ndarray: def resize_image_to_160x160(image_path: str) -> np.ndarray: image = cv2.imread(image_path, cv2.IMREAD_COLOR) image = cv2.cvtColor(image, cv2.COLOR_BGR2RGB) - return _resize_to_160x160(image) + return _resize_to_160x160(image) \ No newline at end of file diff --git a/tests/test_file_utils.py b/tests/test_file_utils.py index d42fda8..4463854 100644 --- a/tests/test_file_utils.py +++ b/tests/test_file_utils.py @@ -1,7 +1,7 @@ import unittest from pathlib import Path -import cv2 +from cv2 import cv2 from app.utils import file_utils @@ -21,7 +21,7 @@ class FileUtilsTestCase(unittest.TestCase): blob = """  """ - self.assertTrue(file_utils.write_webcam_blob(blob, 'test1-15922865715-1')) + self.assertTrue(file_utils.write_photo(blob, 'test1-15922865715-1')) def test_resize_image_to_160x160(self): image = str(FILE_PREFIX / 'test1-15922865715-1.jpg') @@ -35,6 +35,6 @@ class FileUtilsTestCase(unittest.TestCase): blob = """  """ - res = file_utils.resize_blob_to_160x160(blob) - cv2.imwrite(str(FILE_PREFIX / 'test1-15922865715-2.jpg'), res) - self.assertEqual((160, 160), res.shape[:2]) + image = file_utils.resize_blob_to_160x160(blob) + cv2.imwrite(str(FILE_PREFIX / 'test1-15922865715-2.jpg'), image) + self.assertEqual((160, 160), image.shape[:2]) -- Gitee