Spaces:
Sleeping
Sleeping
Update app.py
Browse files
app.py
CHANGED
@@ -5,138 +5,134 @@ import requests
|
|
5 |
import tempfile
|
6 |
import mediapipe as mp
|
7 |
import os
|
8 |
-
import
|
9 |
-
import io
|
10 |
-
from PIL import Image
|
11 |
from googleapiclient.discovery import build
|
12 |
from googleapiclient.http import MediaFileUpload
|
13 |
-
from google_auth_oauthlib.flow import InstalledAppFlow
|
14 |
from google.auth.transport.requests import Request
|
15 |
-
import pickle
|
16 |
|
17 |
# === CONFIGURATION ===
|
18 |
API_KEY = "AIzaSyDojJrpauA0XZtCCDUuo9xeQHZQamYKsC4"
|
19 |
CCTVFEED_IDS = ['1KJRkSf2SKEZ1mXS9_si65IwMBtjs6p4n']
|
20 |
REG_FOLDER_ID = '1qkcR7nQTEtiMH9OFUv2bGxVn08E3dKjF'
|
21 |
INTRUDER_FOLDER_ID = '1PPAUWU-wMx7fek73p-hqPqYQypYtG8Ob'
|
22 |
-
|
23 |
|
24 |
# === SETUP ===
|
25 |
mp_face_detection = mp.solutions.face_detection
|
26 |
mp_drawing = mp.solutions.drawing_utils
|
27 |
face_detector = mp_face_detection.FaceDetection(model_selection=0, min_detection_confidence=0.5)
|
28 |
|
29 |
-
|
30 |
-
|
31 |
-
creds = None
|
32 |
-
if os.path.exists('token.pickle'):
|
33 |
-
with open('token.pickle', 'rb') as token:
|
34 |
-
creds = pickle.load(token)
|
35 |
-
if creds and creds.valid:
|
36 |
-
return build('drive', 'v3', credentials=creds)
|
37 |
-
return None
|
38 |
|
39 |
-
def
|
40 |
-
|
41 |
-
|
42 |
-
|
43 |
-
file_metadata = {
|
44 |
-
'name': filename,
|
45 |
-
'parents': [folder_id]
|
46 |
-
}
|
47 |
-
media = MediaFileUpload(file_path, resumable=True)
|
48 |
-
service.files().create(body=file_metadata, media_body=media, fields='id').execute()
|
49 |
-
return True
|
50 |
|
51 |
-
|
52 |
-
def get_drive_links(folder_id, mime_prefix="video/"):
|
53 |
api_url = f"https://www.googleapis.com/drive/v3/files?q='{folder_id}'+in+parents&key={API_KEY}&fields=files(id,name,mimeType)"
|
54 |
-
|
55 |
-
|
56 |
-
|
57 |
-
|
58 |
-
|
59 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
60 |
|
61 |
-
def
|
62 |
-
|
63 |
temp = tempfile.NamedTemporaryFile(delete=False, suffix=".mp4")
|
64 |
-
for chunk in
|
65 |
temp.write(chunk)
|
66 |
temp.close()
|
67 |
return temp.name
|
68 |
|
69 |
-
def
|
70 |
-
|
71 |
-
|
72 |
-
|
73 |
-
|
74 |
-
|
75 |
-
|
76 |
-
|
77 |
-
|
78 |
-
|
79 |
-
|
80 |
-
|
81 |
-
|
82 |
-
|
83 |
-
|
84 |
-
|
|
|
|
|
|
|
|
|
85 |
while cap.isOpened():
|
86 |
ret, frame = cap.read()
|
87 |
if not ret:
|
88 |
break
|
89 |
-
|
90 |
-
frame = cv2.resize(frame, (640, 480))
|
91 |
rgb = cv2.cvtColor(frame, cv2.COLOR_BGR2RGB)
|
92 |
results = face_detector.process(rgb)
|
93 |
-
|
94 |
if results.detections:
|
95 |
-
for
|
96 |
-
|
97 |
ih, iw, _ = frame.shape
|
98 |
-
x, y, w, h = int(
|
99 |
-
|
100 |
-
face
|
101 |
-
|
102 |
-
|
103 |
-
|
104 |
-
|
105 |
-
|
106 |
-
|
107 |
-
|
108 |
-
|
109 |
-
|
110 |
-
if success:
|
111 |
-
st.sidebar.image(face, caption="π¨ Intruder", width=120)
|
112 |
-
else:
|
113 |
-
st.warning("β Upload failed.")
|
114 |
-
else:
|
115 |
-
st.sidebar.image(face, caption="β Duplicate", width=120)
|
116 |
-
detected += 1
|
117 |
cap.release()
|
118 |
-
return
|
119 |
|
120 |
# === STREAMLIT UI ===
|
121 |
-
st.set_page_config(
|
122 |
-
st.title("π Intruder Detection
|
123 |
-
|
124 |
-
|
125 |
-
|
126 |
-
|
127 |
-
|
128 |
-
|
129 |
-
|
130 |
-
|
131 |
-
|
132 |
-
|
133 |
-
|
134 |
-
|
135 |
-
|
136 |
-
|
137 |
-
|
138 |
-
|
139 |
-
|
140 |
-
|
141 |
-
|
142 |
-
|
|
|
|
|
|
|
|
5 |
import tempfile
|
6 |
import mediapipe as mp
|
7 |
import os
|
8 |
+
import pickle
|
|
|
|
|
9 |
from googleapiclient.discovery import build
|
10 |
from googleapiclient.http import MediaFileUpload
|
|
|
11 |
from google.auth.transport.requests import Request
|
|
|
12 |
|
13 |
# === CONFIGURATION ===
|
14 |
API_KEY = "AIzaSyDojJrpauA0XZtCCDUuo9xeQHZQamYKsC4"
|
15 |
CCTVFEED_IDS = ['1KJRkSf2SKEZ1mXS9_si65IwMBtjs6p4n']
|
16 |
REG_FOLDER_ID = '1qkcR7nQTEtiMH9OFUv2bGxVn08E3dKjF'
|
17 |
INTRUDER_FOLDER_ID = '1PPAUWU-wMx7fek73p-hqPqYQypYtG8Ob'
|
18 |
+
MSE_THRESHOLD = 1000
|
19 |
|
20 |
# === SETUP ===
|
21 |
mp_face_detection = mp.solutions.face_detection
|
22 |
mp_drawing = mp.solutions.drawing_utils
|
23 |
face_detector = mp_face_detection.FaceDetection(model_selection=0, min_detection_confidence=0.5)
|
24 |
|
25 |
+
def mse(img1, img2):
|
26 |
+
return np.mean((img1.astype("float") - img2.astype("float")) ** 2)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
27 |
|
28 |
+
def download_image(url):
|
29 |
+
resp = requests.get(url, stream=True)
|
30 |
+
arr = np.asarray(bytearray(resp.content), dtype=np.uint8)
|
31 |
+
return cv2.imdecode(arr, cv2.IMREAD_COLOR)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
32 |
|
33 |
+
def get_faces_from_drive(folder_id):
|
|
|
34 |
api_url = f"https://www.googleapis.com/drive/v3/files?q='{folder_id}'+in+parents&key={API_KEY}&fields=files(id,name,mimeType)"
|
35 |
+
results = requests.get(api_url).json().get("files", [])
|
36 |
+
faces = []
|
37 |
+
for f in results:
|
38 |
+
if f["mimeType"].startswith("image/"):
|
39 |
+
url = f"https://drive.google.com/uc?id={f['id']}&export=download"
|
40 |
+
image = download_image(url)
|
41 |
+
rgb = cv2.cvtColor(image, cv2.COLOR_BGR2RGB)
|
42 |
+
detections = face_detector.process(rgb)
|
43 |
+
if detections.detections:
|
44 |
+
for d in detections.detections:
|
45 |
+
r = d.location_data.relative_bounding_box
|
46 |
+
ih, iw, _ = image.shape
|
47 |
+
x, y, w, h = int(r.xmin * iw), int(r.ymin * ih), int(r.width * iw), int(r.height * ih)
|
48 |
+
face = image[y:y+h, x:x+w]
|
49 |
+
if face.size:
|
50 |
+
face = cv2.resize(face, (100, 100))
|
51 |
+
faces.append(face)
|
52 |
+
return faces
|
53 |
+
|
54 |
+
def get_video_links(folder_id):
|
55 |
+
api_url = f"https://www.googleapis.com/drive/v3/files?q='{folder_id}'+in+parents&key={API_KEY}&fields=files(id,name,mimeType)"
|
56 |
+
results = requests.get(api_url).json().get("files", [])
|
57 |
+
return [(f["name"], f"https://drive.google.com/uc?id={f['id']}&export=download")
|
58 |
+
for f in results if f["mimeType"].startswith("video/")]
|
59 |
|
60 |
+
def download_video(link):
|
61 |
+
resp = requests.get(link, stream=True)
|
62 |
temp = tempfile.NamedTemporaryFile(delete=False, suffix=".mp4")
|
63 |
+
for chunk in resp.iter_content(1024*1024):
|
64 |
temp.write(chunk)
|
65 |
temp.close()
|
66 |
return temp.name
|
67 |
|
68 |
+
def upload_intruder(img):
|
69 |
+
creds = None
|
70 |
+
if os.path.exists("token.pickle"):
|
71 |
+
with open("token.pickle", "rb") as token:
|
72 |
+
creds = pickle.load(token)
|
73 |
+
if not creds or not creds.valid:
|
74 |
+
if creds and creds.expired and creds.refresh_token:
|
75 |
+
creds.refresh(Request())
|
76 |
+
service = build('drive', 'v3', credentials=creds)
|
77 |
+
temp = tempfile.NamedTemporaryFile(delete=False, suffix=".jpg")
|
78 |
+
cv2.imwrite(temp.name, img)
|
79 |
+
file_metadata = {'name': os.path.basename(temp.name), 'parents': [INTRUDER_FOLDER_ID]}
|
80 |
+
media = MediaFileUpload(temp.name, mimetype='image/jpeg')
|
81 |
+
service.files().create(body=file_metadata, media_body=media).execute()
|
82 |
+
os.remove(temp.name)
|
83 |
+
|
84 |
+
def process_video(path, reg_faces, intruder_gallery):
|
85 |
+
cap = cv2.VideoCapture(path)
|
86 |
+
uploaded_faces = []
|
87 |
+
uploaded = 0
|
88 |
while cap.isOpened():
|
89 |
ret, frame = cap.read()
|
90 |
if not ret:
|
91 |
break
|
|
|
|
|
92 |
rgb = cv2.cvtColor(frame, cv2.COLOR_BGR2RGB)
|
93 |
results = face_detector.process(rgb)
|
|
|
94 |
if results.detections:
|
95 |
+
for d in results.detections:
|
96 |
+
r = d.location_data.relative_bounding_box
|
97 |
ih, iw, _ = frame.shape
|
98 |
+
x, y, w, h = int(r.xmin * iw), int(r.ymin * ih), int(r.width * iw), int(r.height * ih)
|
99 |
+
face = frame[y:y+h, x:x+w]
|
100 |
+
if face.size:
|
101 |
+
face_resized = cv2.resize(face, (100, 100))
|
102 |
+
if not any(mse(face_resized, r) < MSE_THRESHOLD for r in reg_faces + uploaded_faces):
|
103 |
+
uploaded_faces.append(face_resized)
|
104 |
+
intruder_gallery.image(face, caption="π New Intruder", use_column_width=True)
|
105 |
+
try:
|
106 |
+
upload_intruder(face)
|
107 |
+
uploaded += 1
|
108 |
+
except:
|
109 |
+
st.warning("Upload failed.")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
110 |
cap.release()
|
111 |
+
return uploaded
|
112 |
|
113 |
# === STREAMLIT UI ===
|
114 |
+
st.set_page_config(layout="wide")
|
115 |
+
st.title("π Intruder Detection (MediaPipe + Google Drive)")
|
116 |
+
sidebar = st.sidebar
|
117 |
+
sidebar.header("π§ Registered Faces")
|
118 |
+
registered_faces = get_faces_from_drive(REG_FOLDER_ID)
|
119 |
+
for idx, face in enumerate(registered_faces):
|
120 |
+
sidebar.image(cv2.cvtColor(face, cv2.COLOR_BGR2RGB), caption=f"Reg {idx+1}", use_column_width=True)
|
121 |
+
|
122 |
+
intruder_gallery = st.sidebar.container()
|
123 |
+
intruder_gallery.header("π¨ Intruders")
|
124 |
+
|
125 |
+
total_uploaded = 0
|
126 |
+
for folder_id in CCTVFEED_IDS:
|
127 |
+
videos = get_video_links(folder_id)
|
128 |
+
for vname, link in videos:
|
129 |
+
st.subheader(f"π₯ {vname}")
|
130 |
+
with st.spinner("Downloading video..."):
|
131 |
+
video_file = download_video(link)
|
132 |
+
with st.spinner("Detecting faces..."):
|
133 |
+
uploaded = process_video(video_file, registered_faces, intruder_gallery)
|
134 |
+
st.success(f"β
Uploaded {uploaded} intruder(s) from {vname}")
|
135 |
+
total_uploaded += uploaded
|
136 |
+
os.remove(video_file)
|
137 |
+
|
138 |
+
st.info(f"π― Done. Total intruders uploaded: {total_uploaded}")
|