Soham Chandratre
minor changes
aafa470
raw
history blame
11.5 kB
from fastapi import APIRouter, HTTPException,Depends,File, UploadFile
from fastapi.responses import JSONResponse
from config.database import admin_collection, user_collection,notification_collection,pothole_image_collection
from model.pothole_model import load_image_model
from utils.auth import create_access_token, hash_password, verify_password, verify_token
from fastapi.security import HTTPBearer, HTTPAuthorizationCredentials
from schema.model import Admin, PoholeInfo, PotInfoById, PotholeFilters, PotholeModel, UpdatePotholeInfo, User, UserLogin, VerifyOtp
from utils.email_validator import send_activation_email
import requests
import uuid
from bson import ObjectId
import random
security_scheme = HTTPBearer()
router = APIRouter()
activation_tokens={}
import firebase_admin
from firebase_admin import credentials, storage
# Initialize Firebase Admin SDK
cred = credentials.Certificate("pothole-detection-c31a0-firebase-adminsdk-xirxs-4ad5f554ca.json")
firebase_admin.initialize_app(cred, {
'storageBucket': 'pothole-detection-c31a0.appspot.com'
})
def generate_otp():
return str(random.randint(100000, 999999))
#----------------------------------------Admin Collections----------------------------------------
# Admin Registration Api
@router.post("/api/admin/registerAdmin", tags=["admin"])
async def register_admin(admin: Admin):
hashed_password = hash_password(admin.password)
admin_dict = admin.dict()
admin_dict['password'] = hashed_password
activation_otp = generate_otp()
admin_dict['otp'] = activation_otp
result = admin_collection.insert_one(admin_dict)
return {"message": "Admin created successfully", "user_id": str(result.inserted_id)}
#Api to get all admins
# @router.get("/api/admins/getAllAdmin", tags=["admin"])
# async def get_all_admins():
# admins = admin_collection.find({})
# admin_list = []
# for admin in admins:
# admin["_id"] = str(admin["_id"])
# admin_list.append(admin)
# return admin_list
#-------------------------------------------User Collections-------------------------------------------
# User Login API
@router.post("/api/user/userLogin", tags=["user"])
async def user_login(userLogin: UserLogin):
# Check if the provided username and password are for admin
if userLogin.userName == "admin001" and userLogin.password == "Admin@123":
collection = admin_collection
else:
collection = user_collection
user = collection.find_one({"userName": userLogin.userName})
if not user or not verify_password(userLogin.password, user.get('password', '')):
raise HTTPException(status_code=401, detail="Incorrect username or password")
# Check if the user is verified
if not user.get('isVerified'):
raise HTTPException(status_code=403, detail="Account is not verified")
# Extract additional user data
user_data = {
"email": user.get("email"),
"username": user.get("userName"),
"id": str(user.get("_id")),
"role":user.get("role")
}
token = create_access_token({"sub": userLogin.userName})
response_data = {"access_token": token, "token_type": "bearer", "user_data": user_data}
return response_data
# User Registration Api
@router.post("/api/user/registerUser", tags=["user"])
async def create_user(user: User):
hashed_password = hash_password(user.password)
user_dict = user.dict()
user_dict['password'] = hashed_password
activation_otp = generate_otp()
user_dict['otp'] = activation_otp
# Send activation email
send_activation_email(user.email, activation_otp)
# result = user_collection.insert_one(user_dict)
# return {"message": "User created successfully. Activation email sent.",}
result = user_collection.insert_one(user_dict)
return {"message": "User created successfully", "userId": str(result.inserted_id)}
# Otp Verification Api
@router.post("/api/user/verifyOtp", tags=["user"])
async def verify_otp(verifyOtp:VerifyOtp):
# Check if the user exists with the given email
user = user_collection.find_one({"email": verifyOtp.email})
if user is None:
raise HTTPException(status_code=404, detail="User not found")
# Check if the provided OTP matches the stored OTP
if user.get('otp') != verifyOtp.otp:
raise HTTPException(status_code=400, detail="Invalid OTP")
# Update the user's isVerified field to True
user_collection.update_one({"email": verifyOtp.email}, {"$set": {"isVerified": True}})
return {"message": "OTP verified successfully and user is verified"}
# Api to show the registered users list
@router.get("/api/user/getAllUsers", tags=["user"], dependencies=[Depends(security_scheme)])
async def get_all_user(token: HTTPAuthorizationCredentials = Depends(security_scheme)):
verify_token(token.credentials)
users = user_collection.find({})
user_list = []
for user in users:
user["_id"] = str(user["_id"]) # Convert ObjectId to string
user_list.append(user)
return user_list
#-------------------------------------------Pothole Collections-------------------------------------------
# Api to submit the information about pothole
@router.post("/api/information/submitInformation", tags=["pothole"], dependencies=[Depends(security_scheme)])
async def upload_file(potholeInfo:PoholeInfo,token: HTTPAuthorizationCredentials = Depends(security_scheme)):
verify_token(token.credentials)
pothole_info = potholeInfo.dict()
notification_collection.insert_one(pothole_info)
return {"message": "information submitted sucesssuccessfully"}
# Api to update the information about pothole
@router.put("/api/information/updateInformation", tags=["pothole"], dependencies=[Depends(security_scheme)])
async def update_pothole_information(update_data: UpdatePotholeInfo, token: HTTPAuthorizationCredentials = Depends(security_scheme)):
verify_token(token.credentials)
try:
# Update the pothole information in the collection
result = notification_collection.update_one({"_id": ObjectId(update_data.infoID)}, {"$set": {"status": update_data.status, "assignee": update_data.assignee}})
if result.modified_count == 1:
return {"message": "Pothole information updated successfully"}
else:
return {"message": "No changes were made. Pothole information remains unchanged."}
except Exception as e:
return JSONResponse(content={"message": f"Error occurred: {str(e)}"}, status_code=500)
# @router.get("/api/information/getAllFiles", tags=["pothole"], dependencies=[Depends(security_scheme)])
# async def get_all_user(token: HTTPAuthorizationCredentials = Depends(security_scheme)):
# verify_token(token.credentials)
# files = pothole_image_collection.find({})
# file_list = []
# for file in files:
# file["_id"] = str(file["_id"]) # Convert ObjectId to string
# file_list.append(file)
# return file_list
# Function to upload file to Firebase Storage
import os
async def upload_file_to_firebase(file: UploadFile, token: HTTPAuthorizationCredentials):
verify_token(token.credentials)
try:
# Generate a unique ID for the file
fileID = uuid.uuid4().hex
# Extract the filename from the path provided by Flutter
filename = os.path.basename(file.filename)
# Get reference to Firebase Storage bucket
bucket = storage.bucket()
# Create a blob object with the filename
blob = bucket.blob(fileID + "-" + filename)
# Upload the file
blob.upload_from_file(file.file)
blob.make_public()
# Get the public URL of the uploaded file
url = blob.public_url
# Insert the document into the collection with Firebase URL
pothole_image_collection.insert_one({"fileID": fileID, "url": url, "filename": filename})
return JSONResponse(content={"message": "file uploaded successfully", "file_id": fileID})
except Exception as e:
return JSONResponse(content={"message": f"Error occurred: {str(e)}"}, status_code=500)
@router.post("/api/information/fileUpload", tags=["pothole"])
async def upload_file_api(file: UploadFile = File(...), token: HTTPAuthorizationCredentials = Depends(security_scheme)):
return await upload_file_to_firebase(file, token)
# Api to get all information about pothole
@router.post("/api/information/getAllPotholeInformation", tags=["pothole"], dependencies=[Depends(security_scheme)])
async def get_all_info_with_filters(filters: PotholeFilters, token: HTTPAuthorizationCredentials = Depends(security_scheme)):
verify_token(token.credentials)
# Construct a filter query based on provided parameters
filter_query = {}
if filters.userID == "66142a506d8f2a0f116fd613":
# If userID is the specific ID, do not filter by userID
pass
elif filters.userID:
filter_query["userId"] = filters.userID
# Always apply the status filter
if filters.status:
filter_query["status"] = filters.status
# Apply the filter query to find relevant pothole information
pothole_informations = notification_collection.find(filter_query)
info_list = []
for info in pothole_informations:
info["_id"] = str(info["_id"])
file_id = info.get('fileID')
if file_id:
image_data = pothole_image_collection.find_one({"fileID": file_id})
if image_data:
info['image'] = image_data.get('url')
else:
print("No image data found for file ID:", file_id) # Debugging print
info_list.append(info)
return info_list
# Api to get information about pothole by unique id
@router.post("/api/information/getPotholeInformationById", tags=["pothole"], dependencies=[Depends(security_scheme)])
async def get_data_by_id(potHoleInfoById:PotInfoById,token: HTTPAuthorizationCredentials = Depends(security_scheme)):
verify_token(token.credentials)
try:
object_id = ObjectId(potHoleInfoById.infoID)
data = notification_collection.find_one({"_id": object_id})
if data:
data["_id"] = str(data["_id"])
image_data = pothole_image_collection.find_one({"fileID": data["fileID"]})
if image_data:
data["image"] = image_data["url"] # Append image to the response
return data
else:
raise HTTPException(status_code=404, detail="Data not found")
except:
raise HTTPException(status_code=400, detail="Invalid ID format")
# Api to to verify the given object has pothole or not
@router.post("/api/information/verifyPothole", tags=["pothole"], dependencies=[Depends(security_scheme)])
async def verify_pothole(potholeModel: PotholeModel, token: HTTPAuthorizationCredentials = Depends(security_scheme)):
verify_token(token.credentials)
try:
# Get image bytes from URL
response = requests.get(potholeModel.image)
image_bytes = response.content
# Pass image bytes to your model function
results = load_image_model(image_bytes)
# if results == 1:
# return JSONResponse(content={"response": "Pothole"})
# else:
# return JSONResponse(content={"response": "notPothole"})
return JSONResponse(content={"response": "Pothole"})
except Exception as e:
return JSONResponse(content={"response": f"{e}"})