From 1a825ba46cf2a632b551b1e29ecb85bedfe79d35 Mon Sep 17 00:00:00 2001 From: akashbhome Date: Fri, 27 Mar 2026 14:17:21 +0530 Subject: [PATCH] updated Payment reconcillation code --- .env | 9 + .gitignore | 7 + Dockerfile | 18 + README.md | 0 config.py | 21 + controllers/auth_controller.py | 46 + controllers/block_controller.py | 117 ++ controllers/district_controller.py | 84 + controllers/excel_upload_controller.py | 441 ++++ controllers/gst_release_controller.py | 46 + controllers/hold_types_controller.py | 77 + controllers/invoice_controller.py | 101 + controllers/log_controller.py | 30 + controllers/payment_controller.py | 101 + controllers/pmc_report_controller.py | 40 + controllers/report_controller.py | 67 + controllers/state_controller.py | 69 + controllers/subcontractor_controller.py | 94 + controllers/village_controller.py | 151 ++ docker-compose.yml | 35 + download/Contractor_Report_1.xlsx | Bin 0 -> 8218 bytes logs/activity.log | 2405 ++++++++++++++++++++++ logs/audit.log | 74 + main.py | 68 + model/Auth.py | 63 + model/Block.py | 161 ++ model/ContractorInfo.py | 65 + model/District.py | 80 + model/FolderAndFile.py | 53 + model/GST.py | 51 + model/HoldTypes.py | 90 + model/Invoice.py | 237 +++ model/ItemCRUD.py | 397 ++++ model/Log.py | 84 + model/PmcReport.py | 456 ++++ model/Report.py | 382 ++++ model/State.py | 156 ++ model/Subcontractor.py | 140 ++ model/Utilities.py | 67 + model/Village.py | 186 ++ model/gst_release.py | 160 ++ model/payment.py | 236 +++ requirements.txt | 4 + static/css/base.css | 89 + static/css/index.css | 226 ++ static/css/invoice.css | 395 ++++ static/css/invoice1.css | 313 +++ static/css/report.css | 181 ++ static/css/show_excel.css | 212 ++ static/css/style.css | 448 ++++ static/css/subcontractor_report.css | 199 ++ static/css/upload_excel_file.css | 111 + static/download/Contractor_Report_1.xlsx | Bin 0 -> 6209 bytes static/download/Contractor_Report_2.xlsx | Bin 0 -> 8449 bytes static/download/Contractor_Report_3.xlsx | Bin 0 -> 39071 bytes static/download/Contractor_Report_4.xlsx | Bin 0 -> 9147 bytes static/images/icons/bin_red_icon.png | Bin 0 -> 4624 bytes static/images/icons/pen_blue_icon.png | Bin 0 -> 3015 bytes static/js/block.js | 116 ++ static/js/district.js | 62 + static/js/edit_hold_type.js | 18 + static/js/holdAmount.js | 95 + static/js/hold_types.js | 39 + static/js/invoice.js | 271 +++ static/js/save_data_success.js | 39 + static/js/save_excel_file.js | 21 + static/js/searchContractor.js | 43 + static/js/search_on_table.js | 132 ++ static/js/showSuccessAlert.js | 8 + static/js/sorting.js | 66 + static/js/state.js | 61 + static/js/subcontractor.js | 49 + static/js/validateFileInput.js | 12 + static/js/village.js | 250 +++ templates/activity_log.html | 116 ++ templates/add_block.html | 99 + templates/add_district.html | 85 + templates/add_gst_release.html | 209 ++ templates/add_hold_type.html | 73 + templates/add_invoice.html | 536 +++++ templates/add_payment.html | 207 ++ templates/add_purchase_order.html | 147 ++ templates/add_state.html | 74 + templates/add_subcontractor.html | 127 ++ templates/add_village.html | 129 ++ templates/add_work_order.html | 137 ++ templates/admin_profile.html | 49 + templates/base.html | 97 + templates/edit_block.html | 60 + templates/edit_district.html | 26 + templates/edit_grn.html | 39 + templates/edit_gst_release.html | 45 + templates/edit_hold_type.html | 57 + templates/edit_invoice.html | 226 ++ templates/edit_payment.html | 49 + templates/edit_purchase.html | 47 + templates/edit_state.html | 17 + templates/edit_subcontractor.html | 50 + templates/edit_village.html | 44 + templates/grn_form.html | 77 + templates/index.html | 164 ++ templates/login.html | 90 + templates/pmc_report.html | 340 +++ templates/purchase_order.html | 10 + templates/purchase_order_report.html | 205 ++ templates/report.html | 108 + templates/show_excel_file.html | 85 + templates/subcontractor_report.html | 372 ++++ templates/unreleased_gst.html | 35 + templates/uploadExcelFile.html | 24 + templates/work_order_report.html | 209 ++ unusedCode/unused.py | 482 +++++ unusedCode/workOrder.py | 128 ++ 113 files changed, 15699 insertions(+) create mode 100644 .env create mode 100644 .gitignore create mode 100644 Dockerfile create mode 100644 README.md create mode 100644 config.py create mode 100644 controllers/auth_controller.py create mode 100644 controllers/block_controller.py create mode 100644 controllers/district_controller.py create mode 100644 controllers/excel_upload_controller.py create mode 100644 controllers/gst_release_controller.py create mode 100644 controllers/hold_types_controller.py create mode 100644 controllers/invoice_controller.py create mode 100644 controllers/log_controller.py create mode 100644 controllers/payment_controller.py create mode 100644 controllers/pmc_report_controller.py create mode 100644 controllers/report_controller.py create mode 100644 controllers/state_controller.py create mode 100644 controllers/subcontractor_controller.py create mode 100644 controllers/village_controller.py create mode 100644 docker-compose.yml create mode 100644 download/Contractor_Report_1.xlsx create mode 100644 logs/activity.log create mode 100644 logs/audit.log create mode 100644 main.py create mode 100644 model/Auth.py create mode 100644 model/Block.py create mode 100644 model/ContractorInfo.py create mode 100644 model/District.py create mode 100644 model/FolderAndFile.py create mode 100644 model/GST.py create mode 100644 model/HoldTypes.py create mode 100644 model/Invoice.py create mode 100644 model/ItemCRUD.py create mode 100644 model/Log.py create mode 100644 model/PmcReport.py create mode 100644 model/Report.py create mode 100644 model/State.py create mode 100644 model/Subcontractor.py create mode 100644 model/Utilities.py create mode 100644 model/Village.py create mode 100644 model/gst_release.py create mode 100644 model/payment.py create mode 100644 requirements.txt create mode 100644 static/css/base.css create mode 100644 static/css/index.css create mode 100644 static/css/invoice.css create mode 100644 static/css/invoice1.css create mode 100644 static/css/report.css create mode 100644 static/css/show_excel.css create mode 100644 static/css/style.css create mode 100644 static/css/subcontractor_report.css create mode 100644 static/css/upload_excel_file.css create mode 100644 static/download/Contractor_Report_1.xlsx create mode 100644 static/download/Contractor_Report_2.xlsx create mode 100644 static/download/Contractor_Report_3.xlsx create mode 100644 static/download/Contractor_Report_4.xlsx create mode 100644 static/images/icons/bin_red_icon.png create mode 100644 static/images/icons/pen_blue_icon.png create mode 100644 static/js/block.js create mode 100644 static/js/district.js create mode 100644 static/js/edit_hold_type.js create mode 100644 static/js/holdAmount.js create mode 100644 static/js/hold_types.js create mode 100644 static/js/invoice.js create mode 100644 static/js/save_data_success.js create mode 100644 static/js/save_excel_file.js create mode 100644 static/js/searchContractor.js create mode 100644 static/js/search_on_table.js create mode 100644 static/js/showSuccessAlert.js create mode 100644 static/js/sorting.js create mode 100644 static/js/state.js create mode 100644 static/js/subcontractor.js create mode 100644 static/js/validateFileInput.js create mode 100644 static/js/village.js create mode 100644 templates/activity_log.html create mode 100644 templates/add_block.html create mode 100644 templates/add_district.html create mode 100644 templates/add_gst_release.html create mode 100644 templates/add_hold_type.html create mode 100644 templates/add_invoice.html create mode 100644 templates/add_payment.html create mode 100644 templates/add_purchase_order.html create mode 100644 templates/add_state.html create mode 100644 templates/add_subcontractor.html create mode 100644 templates/add_village.html create mode 100644 templates/add_work_order.html create mode 100644 templates/admin_profile.html create mode 100644 templates/base.html create mode 100644 templates/edit_block.html create mode 100644 templates/edit_district.html create mode 100644 templates/edit_grn.html create mode 100644 templates/edit_gst_release.html create mode 100644 templates/edit_hold_type.html create mode 100644 templates/edit_invoice.html create mode 100644 templates/edit_payment.html create mode 100644 templates/edit_purchase.html create mode 100644 templates/edit_state.html create mode 100644 templates/edit_subcontractor.html create mode 100644 templates/edit_village.html create mode 100644 templates/grn_form.html create mode 100644 templates/index.html create mode 100644 templates/login.html create mode 100644 templates/pmc_report.html create mode 100644 templates/purchase_order.html create mode 100644 templates/purchase_order_report.html create mode 100644 templates/report.html create mode 100644 templates/show_excel_file.html create mode 100644 templates/subcontractor_report.html create mode 100644 templates/unreleased_gst.html create mode 100644 templates/uploadExcelFile.html create mode 100644 templates/work_order_report.html create mode 100644 unusedCode/unused.py create mode 100644 unusedCode/workOrder.py diff --git a/.env b/.env new file mode 100644 index 0000000..5bfeba1 --- /dev/null +++ b/.env @@ -0,0 +1,9 @@ +Secret_Key = 9f2a1b8c4d6e7f0123456789abcdef01 + +MYSQL_HOST=127.0.0.1 +MYSQL_USER=root +MYSQL_PASSWORD=root +MYSQL_DB=test + +DEFAULT_USERNAME=admin +DEFAULT_PASSWORD=admin123 diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..f22dbe7 --- /dev/null +++ b/.gitignore @@ -0,0 +1,7 @@ +venv/ +*.pyc +__pycache__/ + +static/downloads/ +static/uploads/ + diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..411d674 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,18 @@ +# Use official Python image +FROM python:3.9 + +# Set working directory inside container +WORKDIR /app + +# Copy all files to container +COPY . . + +# Install dependencies +RUN pip install --no-cache-dir -r requirements.txt + +# Expose Flask's default port +EXPOSE 5000 + +# Run Flask app +CMD ["python", "main.py"] + diff --git a/README.md b/README.md new file mode 100644 index 0000000..e69de29 diff --git a/config.py b/config.py new file mode 100644 index 0000000..b147757 --- /dev/null +++ b/config.py @@ -0,0 +1,21 @@ +import mysql.connector +import os +from dotenv import load_dotenv + +# Load environment variables from .env file +load_dotenv() + +# Get MySQL credentials from environment variables +MYSQL_HOST = os.getenv("MYSQL_HOST") +MYSQL_USER = os.getenv("MYSQL_USER") +MYSQL_PASSWORD = os.getenv("MYSQL_PASSWORD") +MYSQL_DB = os.getenv("MYSQL_DB") + +# Connect to MySQL +def get_db_connection(): + return mysql.connector.connect( + host=MYSQL_HOST, + user=MYSQL_USER, + password=MYSQL_PASSWORD, + database=MYSQL_DB + ) \ No newline at end of file diff --git a/controllers/auth_controller.py b/controllers/auth_controller.py new file mode 100644 index 0000000..0e63627 --- /dev/null +++ b/controllers/auth_controller.py @@ -0,0 +1,46 @@ +from flask import Blueprint, render_template, request, redirect, url_for, flash, session +from flask_login import login_user, logout_user, login_required, current_user + +from model.Auth import LoginLDAP, User +from model.Log import LogHelper + +auth_bp = Blueprint('auth', __name__) + + +@auth_bp.route('/login', methods=['GET', 'POST']) +def login(): + + if request.method == 'POST': + + loginData = LoginLDAP(request) + + if loginData.isValidLogin: + + if loginData.isDefaultCredentials: + LogHelper.log_action('Login', f"User {loginData.username} logged in (static user)") + else: + LogHelper.log_action('Login', f"User {loginData.username} logged in (LDAP)") + + session['username'] = loginData.username + + login_user(User(loginData.username)) + + return redirect(url_for('index')) + + else: + flash(loginData.errorMessage, 'danger') + + return render_template("login.html") + + +@auth_bp.route('/logout') +@login_required +def logout(): + + LogHelper.log_action('Logout', f"User {current_user.id} logged out") + + logout_user() + + flash('You have been logged out.', 'info') + + return redirect(url_for('auth.login')) \ No newline at end of file diff --git a/controllers/block_controller.py b/controllers/block_controller.py new file mode 100644 index 0000000..8f2e80a --- /dev/null +++ b/controllers/block_controller.py @@ -0,0 +1,117 @@ +import config +from flask import Blueprint, render_template, request, redirect, url_for, jsonify, flash +from flask_login import login_required + +from model.State import State +from model.Block import Block +from model.Utilities import HtmlHelper + +block_bp = Blueprint('block', __name__) + +# --- Add Block page ------- +@block_bp.route('/add_block', methods=['GET', 'POST']) +@login_required +def add_block(): + block = Block() + + if request.method == 'POST': + block.AddBlock(request) + return block.resultMessage + + state = State() + states = state.GetAllStates(request=request) + + block_data = block.GetAllBlocks(request) + + return render_template( + 'add_block.html', + states=states, + block_data=block_data + ) + + +# ✅ NEW ROUTE (FIX FOR DISTRICT FETCH) +@block_bp.route('/get_districts/') +@login_required +def get_districts(state_id): + + connection = config.get_db_connection() + cursor = connection.cursor() + + cursor.callproc("GetDistrictsByStateId", (state_id,)) + + districts = [] + for rs in cursor.stored_results(): + districts = rs.fetchall() + + cursor.close() + connection.close() + + district_list = [] + + for district in districts: + district_list.append({ + "id": district[0], + "name": district[1] + }) + + return jsonify(district_list) + + +@block_bp.route('/check_block', methods=['POST']) +@login_required +def check_block(): + + block = Block() + return block.CheckBlock(request) + + +@block_bp.route('/edit_block/', methods=['GET', 'POST']) +@login_required +def edit_block(block_id): + + block = Block() + + if request.method == 'POST': + block.EditBlock(request, block_id) + block.resultMessage + + if block.resultMessage: + flash("Block updated successfully!", "success") + return redirect(url_for('block.add_block')) + else: + flash(block.resultMessage, "error") + + + connection = config.get_db_connection() + cursor = connection.cursor() + + cursor.callproc("GetAllStates") + for rs in cursor.stored_results(): + states = rs.fetchall() + + cursor.callproc("GetAllDistrictsData") + for rs in cursor.stored_results(): + districts = rs.fetchall() + + block_data = block.GetBlockByID(block_id) + + cursor.close() + connection.close() + + return render_template( + 'edit_block.html', + block_data=block_data, + states=states, + districts=districts + ) + + +@block_bp.route('/delete_block/') +@login_required +def delete_block(block_id): + + block = Block() + block.DeleteBlock(request, block_id) + + return redirect(url_for('block.add_block')) \ No newline at end of file diff --git a/controllers/district_controller.py b/controllers/district_controller.py new file mode 100644 index 0000000..fe6efd7 --- /dev/null +++ b/controllers/district_controller.py @@ -0,0 +1,84 @@ +from flask import Blueprint, render_template, request, redirect, url_for, flash +from flask_login import login_required + +from model.District import District +from model.State import State + +district_bp = Blueprint('district', __name__) + +# ------- District page -------- +@district_bp.route('/add_district', methods=['GET', 'POST']) +@login_required +def add_district(): + district = District() + + if request.method == 'POST': + district.AddDistrict(request=request) + return district.resultMessage + + state = State() + states = state.GetAllStates(request=request) + + districtdata = district.GetAllDistricts(request=request) + + return render_template( + 'add_district.html', + districtdata=districtdata, + states=states + ) + +# ------- District check -------- +@district_bp.route('/check_district', methods=['POST']) +@login_required +def check_district(): + + district = District() + + return district.CheckDistrict(request=request) + +# ------- District delete by district id -------- +@district_bp.route('/delete_district/') +@login_required +def delete_district(district_id): + + district = District() + + district.DeleteDistrict(request=request, district_id=district_id) + + if not district.isSuccess: + return district.resultMessage + else: + return redirect(url_for('district.add_district')) + + +# ------- District update by district id -------- +@district_bp.route('/edit_district/', methods=['GET', 'POST']) +@login_required +def edit_district(district_id): + + district = District() + state = State() + + if request.method == 'POST': + + district.EditDistrict(request=request, district_id=district_id) + + if district.isSuccess: + flash("District updated successfully!", "success") + return redirect(url_for('district.add_district')) + else: + flash(district.resultMessage, "error") + + districtdata = district.GetDistrictByID(request=request, district_id=district_id) + + if not districtdata: + flash("District not found", "error") + return redirect(url_for('district.add_district')) + + states = state.GetAllStates(request=request) + + return render_template( + 'edit_district.html', + districtdata=districtdata, + states=states + ) \ No newline at end of file diff --git a/controllers/excel_upload_controller.py b/controllers/excel_upload_controller.py new file mode 100644 index 0000000..f316ab6 --- /dev/null +++ b/controllers/excel_upload_controller.py @@ -0,0 +1,441 @@ +import config +import ast +import re +import openpyxl +from flask_login import current_user +from flask_login import login_required +from flask import Blueprint, request, render_template, redirect, url_for, jsonify + +from model.Log import LogHelper +from model.FolderAndFile import FolderAndFile + +excel_bp = Blueprint('excel', __name__) + +# ---------------- Upload Excel File ---------------- +@excel_bp.route('/upload_excel_file', methods=['GET', 'POST']) +@login_required +def upload(): + if request.method == 'POST': + file = request.files.get('file') + if file and file.filename.endswith('.xlsx'): + filepath =FolderAndFile.get_upload_path(file.filename) + + file.save(filepath) + + LogHelper.log_action( + "Upload Excel File", + f"User {current_user.id} Upload Excel File '{file.filename}'" + ) + return redirect(url_for('excel.show_table', filename=file.filename)) + + return render_template('uploadExcelFile.html') + + +# ---------------- Show Excel Table ---------------- +@excel_bp.route('/show_table/') +def show_table(filename): + global data + data = [] + + filepath = FolderAndFile.get_upload_path(filename) + wb = openpyxl.load_workbook(filepath, data_only=True) + sheet = wb.active + + file_info = { + "Subcontractor": sheet.cell(row=1, column=2).value, + "State": sheet.cell(row=2, column=2).value, + "District": sheet.cell(row=3, column=2).value, + "Block": sheet.cell(row=4, column=2).value, + } + + errors = [] + subcontractor_data = None + state_data = None + district_data = None + block_data = None + + connection = config.get_db_connection() + if connection: + try: + cursor = connection.cursor(dictionary=True) + + cursor.callproc('GetStateByName', [file_info['State']]) + for result in cursor.stored_results(): + state_data = result.fetchone() + if not state_data: + errors.append(f"State '{file_info['State']}' is not valid. Please add it.") + + if state_data: + cursor.callproc('GetDistrictByNameAndStates', [file_info['District'], state_data['State_ID']]) + for result in cursor.stored_results(): + district_data = result.fetchone() + if not district_data: + errors.append(f"District '{file_info['District']}' is not valid under state '{file_info['State']}'.") + + if district_data: + cursor.callproc('GetBlockByNameAndDistricts', [file_info['Block'], district_data['District_ID']]) + for result in cursor.stored_results(): + block_data = result.fetchone() + if not block_data: + errors.append(f"Block '{file_info['Block']}' is not valid under district '{file_info['District']}'.") + + cursor.callproc('GetSubcontractorByName', [file_info['Subcontractor']]) + for result in cursor.stored_results(): + subcontractor_data = result.fetchone() + + if not subcontractor_data: + cursor.callproc('InsertSubcontractor', [file_info['Subcontractor']]) + connection.commit() + cursor.callproc('GetSubcontractorByName', [file_info['Subcontractor']]) + for result in cursor.stored_results(): + subcontractor_data = result.fetchone() + + cursor.callproc("GetAllHoldTypes") + hold_types_data = [] + for ht in cursor.stored_results(): + hold_types_data = ht.fetchall() + hold_types_lookup = {row['hold_type'].lower(): row['hold_type_id'] for row in hold_types_data if row['hold_type']} + + cursor.close() + except Exception as e: + print(f"Database error: {e}") + return f"Database operation failed: {e}", 500 + finally: + connection.close() + + variables = {} + hold_columns = [] + hold_counter = 0 + + for j in range(1, sheet.max_column + 1): + col_value = sheet.cell(row=5, column=j).value + if col_value: + variables[col_value] = j + if 'hold' in str(col_value).lower(): + hold_counter += 1 + hold_type_key = str(col_value).lower().strip() + hold_type_id = hold_types_lookup.get(hold_type_key) + hold_columns.append({ + 'column_name': col_value, + 'column_number': j, + 'hold_type_id': hold_type_id + }) + + for i in range(6, sheet.max_row + 1): + row_data = {} + if sheet.cell(row=i, column=1).value: + row_data["Row Number"] = i + for var_name, col_num in variables.items(): + row_data[var_name] = sheet.cell(row=i, column=col_num).value + if sum(1 for value in row_data.values() if value) >= 4: + data.append(row_data) + + for hold in hold_columns: + if hold['hold_type_id']: + print(f" if Column: {hold['column_name']}, Column Number: {hold['column_number']}, Hold Type ID: {hold['hold_type_id']}") + else: + errors.append(f"Hold Type not added ! Column name '{hold['column_name']}'.") + print(f" else Column: {hold['column_name']}, Column Number: {hold['column_number']}, Hold Type ID: {hold['hold_type_id']}") + + return render_template( + 'show_excel_file.html', + file_info=file_info, + variables=variables, + data=data, + subcontractor_data=subcontractor_data, + state_data=state_data, + district_data=district_data, + block_data=block_data, + errors=errors, + hold_columns=hold_columns, + hold_counter=hold_counter + ) + + + # save Excel data +@excel_bp.route('/save_data', methods=['POST']) +def save_data(): + # Extract form data + subcontractor_id = request.form.get("subcontractor_data") + state_id = request.form.get("state_data") + district_id = request.form.get("district_data") + block_id = request.form.get("block_data") + variables = request.form.getlist('variables[]') + hold_columns = request.form.get("hold_columns") + hold_counter = request.form.get("hold_counter") + if not data: + return jsonify({"error": "No data provided to save"}), 400 + if data: + connection = config.get_db_connection() + cursor = connection.cursor() + try: + for entry in data: + save_data = { + "PMC_No": entry.get("PMC_No"), + "Invoice_Details": entry.get("Invoice_Details", ''), + "Work_Type": 'none', + "Invoice_Date": entry.get("Invoice_Date").strftime('%Y-%m-%d') if entry.get( + "Invoice_Date") else None, + "Invoice_No": entry.get("Invoice_No", ''), + "Basic_Amount": entry.get("Basic_Amount", 0.00), + "Debit_Amount": entry.get("Debit_Amount", 0.00), + "After_Debit_Amount": entry.get("After_Debit_Amount", 0.00), + "Amount": entry.get("Amount", 0.00), + "GST_Amount": entry.get("GST_Amount", 0.00), + "TDS_Amount": entry.get("TDS_Amount", 0.00), + "SD_Amount": entry.get("SD_Amount", 0.00), + "On_Commission": entry.get("On_Commission", 0.00), + "Hydro_Testing": entry.get("Hydro_Testing", 0.00), + "Hold_Amount": 0, + "GST_SD_Amount": entry.get("GST_SD_Amount", 0.00), + "Final_Amount": entry.get("Final_Amount", 0.00), + "Payment_Amount": entry.get("Payment_Amount", 0.00), + "Total_Amount": entry.get("Total_Amount", 0.00), + "TDS_Payment_Amount": entry.get("TDS_Payment_Amount", 0.00), + "UTR": entry.get("UTR", ''), + } + village_name, work_type = None, None + village_id = 0 + LogHelper.log_action("Data saved", f"User {current_user.id} Data saved'{ village_name}'") + PMC_No = save_data.get('PMC_No') + Invoice_Details = save_data.get('Invoice_Details') + Invoice_Date = save_data.get('Invoice_Date') + Invoice_No = save_data.get('Invoice_No') + Basic_Amount = save_data.get('Basic_Amount') + Debit_Amount = save_data.get('Debit_Amount') + After_Debit_Amount = save_data.get('After_Debit_Amount') + Amount = save_data.get('Amount') + GST_Amount = save_data.get('GST_Amount') + TDS_Amount = save_data.get('TDS_Amount') + SD_Amount = save_data.get('SD_Amount') + On_Commission = save_data.get('On_Commission') + Hydro_Testing = save_data.get('Hydro_Testing') + GST_SD_Amount = save_data.get('GST_SD_Amount') + Final_Amount = save_data.get('Final_Amount') + Payment_Amount = save_data.get('Payment_Amount') + Total_Amount = save_data.get('Total_Amount') + TDS_Payment_Amount = save_data.get('TDS_Payment_Amount') + UTR = save_data.get('UTR') + + if Invoice_Details: + words = Invoice_Details.lower().split() + if 'village' in words: + village_pos = words.index('village') + village_name = " ".join(words[:village_pos]) + if 'work' in words: + work_pos = words.index('work') + if village_name: + work_type = " ".join(words[village_pos + 1:work_pos + 1]) + else: + work_type = " ".join(words[:work_pos + 1]) + if Invoice_Details and 'village' in Invoice_Details.lower() and 'work' in Invoice_Details.lower(): + # print("village_name ::", village_name, "|| work_type ::", work_type) + if block_id and village_name: + village_id = None + cursor.callproc("GetVillageId", (block_id, village_name)) + for result in cursor.stored_results(): + result = result.fetchone() + village_id = result[0] if result else None + if not village_id: + cursor.callproc("SaveVillage", (village_name, block_id)) + cursor.callproc("GetVillageId", (block_id, village_name)) + for result in cursor.stored_results(): + result = result.fetchone() + village_id = result[0] if result else None + # print("village_id :", village_id) + # print("block_id :", block_id) + # print("invoice :", PMC_No, village_id, work_type, Invoice_Details, Invoice_Date, Invoice_No, + # Basic_Amount, Debit_Amount, After_Debit_Amount, Amount, GST_Amount, TDS_Amount, + # SD_Amount, On_Commission, Hydro_Testing, GST_SD_Amount, Final_Amount) + + args = ( + PMC_No, village_id, work_type, Invoice_Details, Invoice_Date, Invoice_No, + Basic_Amount, Debit_Amount, After_Debit_Amount, Amount, GST_Amount, TDS_Amount, + SD_Amount, On_Commission, Hydro_Testing, GST_SD_Amount, Final_Amount, + subcontractor_id, 0 + ) + + # print("All invoice Details ",args) + # add subcontarctor id in invoice table + results = cursor.callproc('SaveInvoice', args) + invoice_id = results[-1] + + + cursor.callproc( + "SavePayment", + ( + PMC_No, + Invoice_No, # required + Payment_Amount, + TDS_Payment_Amount, + Total_Amount, + UTR, + invoice_id # last + ) + ) + + # print("invoice id from the excel ", invoice_id) + if isinstance(hold_columns, str): + hold_columns = ast.literal_eval(hold_columns) + if isinstance(hold_columns, list) and all(isinstance(hold, dict) for hold in hold_columns): + for hold in hold_columns: + # print(f"Processing hold: {hold}") + hold_column_name = hold.get('column_name') # Get column name + hold_type_id = hold.get('hold_type_id') # Get hold_type_id + if hold_column_name: + hold_amount = entry.get( + hold_column_name) # Get the value for that specific hold column + if hold_amount is not None: + # print(f"Processing hold type: {hold_column_name}, Hold Amount: {hold_amount}") + hold_join_data = { + "Contractor_Id": subcontractor_id, + "Invoice_Id": invoice_id, + "hold_type_id": hold_type_id, + "hold_amount": hold_amount + } + cursor.callproc('InsertHoldJoinData', [ + hold_join_data['Contractor_Id'], hold_join_data['Invoice_Id'], + hold_join_data['hold_type_id'], hold_join_data['hold_amount'] + ]) + connection.commit() + print(f"Inserted hold join data: {hold_join_data}") + else: + print(f"Invalid hold entry: {hold}") + else: + print("Hold columns data is not a valid list of dictionaries.") +#---------------------------------------------Credit Note--------------------------------------------------------------------------- + elif any(keyword in Invoice_Details.lower() for keyword in ['credit note','logging report']): + # print("Credit note found:", PMC_No, Invoice_No, Basic_Amount, Debit_Amount, Final_Amount, + # After_Debit_Amount, GST_Amount, Amount, Final_Amount, Payment_Amount, Total_Amount, UTR, Invoice_No) + cursor.callproc( + 'AddCreditNoteFromExcel', + [ + PMC_No, Invoice_Details, Basic_Amount, Debit_Amount, After_Debit_Amount, + GST_Amount, Amount, Final_Amount, Payment_Amount, Total_Amount, UTR, + subcontractor_id, Invoice_No + ] + ) +#-----------------------------------------------Hold Amount---------------------------------------------------------------------- + # Step 1: Normalize Invoice_Details: lowercase, trim, remove extra spaces + normalized_details = re.sub(r'\s+', ' ', Invoice_Details.strip()).lower() + # Step 2: Define lowercase keywords + keywords = [ + 'excess hold', + 'ht', + 'hold release amount', + 'dpr excess hold amount', + 'excess hold amount', + 'Multi to Single layer bill', + 'hold amount', + 'logging report' + ] + # Step 3: Matching condition + if any(kw in normalized_details for kw in keywords): + # print("✅ Match found. Inserting hold release for:", Invoice_Details) + cursor.callproc( + 'AddHoldReleaseFromExcel', + [PMC_No, Invoice_No, Invoice_Details, Basic_Amount, Final_Amount, UTR, subcontractor_id] +) + connection.commit() + # print("✅ Hold release inserted for:", PMC_No, Invoice_Details) + #------------------------------------------------------------------------------------------------------------------ + elif Invoice_Details and any( + keyword in Invoice_Details.lower() for keyword in ['gst', 'release', 'gst release note']): + # print("Gst rels :", PMC_No, Invoice_No, Basic_Amount, Final_Amount,Total_Amount,UTR, subcontractor_id) + cursor.callproc( + 'AddGSTReleaseFromExcel', + [PMC_No, Invoice_No, Basic_Amount, Final_Amount, Total_Amount, UTR, subcontractor_id] + ) + +# -------------------------------------- +# If no village/work detected, only PMC/Payment + if not (Invoice_Details and 'village' in Invoice_Details.lower() and 'work' in Invoice_Details.lower()): + # ---------- Only PMC / Payment rows ---------- + if PMC_No and not Invoice_No and UTR : + # print("No village/work, using PMC only :", PMC_No) + + # check invoice exists + # cursor.execute( + # "SELECT invoice_id FROM invoice WHERE PMC_No=%s ORDER BY invoice_id DESC LIMIT 1", + # (PMC_No,) + # ) + # row = cursor.fetchone() + # invoice_id = row[0] if row else None + + # # insert invoice if not exists + + # if not invoice_id: + print(" extra payment :", PMC_No,Total_Amount,UTR, subcontractor_id) + + cursor.execute( + """ + INSERT INTO invoice (PMC_No,Contractor_Id) VALUES (%s, %s); + """, + (PMC_No, subcontractor_id) + ) + connection.commit() + + cursor.execute( + "SELECT invoice_id FROM invoice WHERE PMC_No=%s AND Contractor_Id =%s ORDER BY invoice_id DESC LIMIT 1", + (PMC_No, subcontractor_id) + ) + row = cursor.fetchone() + invoice_id = row[0] if row else None + + # insert payment + cursor.callproc( + "SavePayment", + ( + PMC_No, + Invoice_No, # required + Payment_Amount, + TDS_Payment_Amount, + Total_Amount, + UTR, + invoice_id + ) + ) + + # if PMC_No and Total_Amount and UTR: + # print("Payment :", PMC_No, Invoice_No, Payment_Amount, TDS_Payment_Amount, Total_Amount, UTR ) + + # Add inoice id in payment table + # cursor.callproc("SavePayment",(PMC_No, Invoice_No, Payment_Amount, TDS_Payment_Amount, Total_Amount, UTR, invoice_id)) + + if not village_id: + village_id = None + # cursor.callproc('InsertOrUpdateInPayment', ( + # PMC_No, + # village_id, + # work_type, + # Invoice_Details, + # Invoice_Date, + # Invoice_No, + # Basic_Amount, + # Debit_Amount, + # After_Debit_Amount, + # Amount, + # GST_Amount, + # TDS_Amount, + # SD_Amount, + # On_Commission, + # Hydro_Testing, + # 0, + # GST_SD_Amount, + # Final_Amount, + # Payment_Amount, + # TDS_Payment_Amount, + # Total_Amount, + # UTR,f + # subcontractor_id + # )) + connection.commit() + return jsonify({"success": "Data saved successfully!"}), 200 + except Exception as e: + connection.rollback() + return jsonify({"error": f"An unexpected error occurred: {e}"}), 500 + finally: + cursor.close() + connection.close() + return render_template('index.html') +# ---------------------- Report -------------------------------- \ No newline at end of file diff --git a/controllers/gst_release_controller.py b/controllers/gst_release_controller.py new file mode 100644 index 0000000..5827f8a --- /dev/null +++ b/controllers/gst_release_controller.py @@ -0,0 +1,46 @@ +# routes/gst_release_routes.py +from flask import Blueprint, render_template, request, redirect, url_for, flash +from flask_login import login_required +from model.gst_release import GSTRelease +from model.Log import LogHelper + +gst_release_bp = Blueprint('gst_release_bp', __name__) +gst_service = GSTRelease() + +# ---------------- ADD GST RELEASE ---------------- +@gst_release_bp.route('/add_gst_release', methods=['GET', 'POST']) +@login_required +def add_gst_release(): + if request.method == 'POST': + gst_service.AddGSTRelease(request) + LogHelper.log_action("Add GST Release", "User added GST release") + flash(gst_service.resultMessage, 'success' if gst_service.isSuccess else 'error') + return redirect(url_for('gst_release_bp.add_gst_release')) + + gst_releases = gst_service.GetAllGSTReleases() + return render_template('add_gst_release.html', gst_releases=gst_releases) + +# ---------------- EDIT GST RELEASE ---------------- +@gst_release_bp.route('/edit_gst_release/', methods=['GET', 'POST']) +@login_required +def edit_gst_release(gst_release_id): + gst_data = gst_service.GetGSTReleaseByID(gst_release_id) + if not gst_data: + return "GST Release not found", 404 + + if request.method == 'POST': + gst_service.EditGSTRelease(request, gst_release_id) + LogHelper.log_action("Edit GST Release", "User edited GST release") + flash(gst_service.resultMessage, 'success' if gst_service.isSuccess else 'error') + return redirect(url_for('gst_release_bp.add_gst_release')) + + return render_template('edit_gst_release.html', gst_release_data=gst_data) + +# ---------------- DELETE GST RELEASE ---------------- +@gst_release_bp.route('/delete_gst_release/', methods=['GET', 'POST']) +@login_required +def delete_gst_release(gst_release_id): + gst_service.DeleteGSTRelease(gst_release_id) + LogHelper.log_action("Delete GST Release", "User deleted GST release") + flash(gst_service.resultMessage, 'success' if gst_service.isSuccess else 'error') + return redirect(url_for('gst_release_bp.add_gst_release')) \ No newline at end of file diff --git a/controllers/hold_types_controller.py b/controllers/hold_types_controller.py new file mode 100644 index 0000000..40b2b20 --- /dev/null +++ b/controllers/hold_types_controller.py @@ -0,0 +1,77 @@ +from flask import Blueprint, render_template, request, jsonify, redirect, url_for +from flask_login import login_required +from model.HoldTypes import HoldTypes +from model.GST import GST + +hold_bp = Blueprint("hold_types", __name__) + + +# ---------------- ADD HOLD TYPE ---------------- +@hold_bp.route('/add_hold_type', methods=['GET','POST']) +@login_required +def add_hold_type(): + hold = HoldTypes() + + if request.method == 'POST': + hold.AddHoldType(request) + # ✅ Always redirect to same page (NO JSON) + return redirect(url_for("hold_types.add_hold_type")) + + # GET request → show data + hold_types = hold.GetAllHoldTypes() + + return render_template( + "add_hold_type.html", + Hold_Types_data=hold_types + ) + +# ---------------- CHECK HOLD TYPE (OPTIONAL LIKE BLOCK) ---------------- +@hold_bp.route('/check_hold_type', methods=['POST']) +@login_required +def check_hold_type(): + + hold = HoldTypes() + return hold.CheckHoldType(request) # if exists + + +# ---------------- EDIT HOLD TYPE ---------------- +@hold_bp.route('/edit_hold_type/', methods=['GET','POST']) +@login_required +def edit_hold_type(id): + + hold = HoldTypes() + + if request.method == 'POST': + hold.EditHoldType(request, id) # ✅ + return hold.resultMessage + + hold_data = hold.GetHoldTypeByID(id) # ✅ + + return render_template( + "edit_hold_type.html", + hold_type=hold_data + ) + + +# ---------------- DELETE HOLD TYPE ---------------- +@hold_bp.route('/delete_hold_type/') +@login_required +def delete_hold_type(id): + + hold = HoldTypes() + hold.DeleteHoldType(request, id) # ✅ + + return redirect(url_for("hold_types.add_hold_type")) + + +# ---------------- GST ---------------- +@hold_bp.route('/unreleased_gst') +@login_required +def unreleased_gst(): + + data = GST.get_unreleased_gst() + + return render_template( + "unreleased_gst.html", + data=data + ) \ No newline at end of file diff --git a/controllers/invoice_controller.py b/controllers/invoice_controller.py new file mode 100644 index 0000000..0beaaf1 --- /dev/null +++ b/controllers/invoice_controller.py @@ -0,0 +1,101 @@ + + + +# controllers/invoice_controller.py + +from flask import Blueprint, request, jsonify, render_template +from flask_login import login_required, current_user +from model.Invoice import * +from model.Log import LogHelper + +invoice_bp = Blueprint('invoice', __name__) + +# ------------------------------- Helpers ------------------------------- +def handle_exception(func): + """Decorator to handle exceptions and return JSON error responses.""" + def wrapper(*args, **kwargs): + try: + return func(*args, **kwargs) + except Exception as e: + return jsonify({"status": "error", "message": str(e)}), 500 + wrapper.__name__ = func.__name__ + return wrapper + +def log_action(action: str, detail: str): + LogHelper.log_action(action, f"User {current_user.id} {detail}") + + +# ------------------------------- Add Invoice ------------------------------- +@invoice_bp.route('/add_invoice', methods=['GET', 'POST']) +@login_required +@handle_exception +def add_invoice(): + if request.method == 'POST': + data = request.form + village_name = data.get('village') + village_result = get_village_id(village_name) + + if not village_result: + return jsonify({"status": "error", "message": f"Village '{village_name}' not found"}), 400 + + village_id = village_result['Village_Id'] + invoice_id = insert_invoice(data, village_id) + assign_subcontractor(data, village_id) + insert_hold_types(data, invoice_id) + + log_action("Add invoice", f"added invoice '{data.get('pmc_no')}'") + return jsonify({"status": "success", "message": "Invoice added successfully"}), 201 + + invoices = get_all_invoice_details() + villages = get_all_villages() + return render_template('add_invoice.html', invoices=invoices, villages=villages) + + +# ------------------------------- Search Subcontractor ------------------------------- +@invoice_bp.route('/search_subcontractor', methods=['POST']) +@login_required +@handle_exception +def search_subcontractor(): + query = request.form.get("query", "").strip() + results = search_contractors(query) + + if not results: + return "
  • No subcontractor found
  • " + + return "".join(f"
  • {r['Contractor_Name']}
  • " for r in results) + + +# ------------------------------- Get Hold Types ------------------------------- +@invoice_bp.route('/get_hold_types', methods=['GET']) +@login_required +@handle_exception +def get_hold_types(): + hold_types = get_all_hold_types() + log_action("Get hold type", f"retrieved hold types '{hold_types}'") + return jsonify(hold_types) + + +# ------------------------------- Edit Invoice ------------------------------- +@invoice_bp.route('/edit_invoice/', methods=['GET', 'POST']) +@login_required +@handle_exception +def edit_invoice(invoice_id): + if request.method == 'POST': + data = request.form + update_invoice(data, invoice_id) + # update_inpayment(data) + log_action("Edit invoice", f"edited invoice '{invoice_id}'") + return jsonify({"status": "success", "message": "Invoice updated successfully"}), 200 + + invoice = get_invoice_by_id(invoice_id) + return render_template('edit_invoice.html', invoice=invoice) + + +# ------------------------------- Delete Invoice ------------------------------- +@invoice_bp.route('/delete_invoice/', methods=['GET']) +@login_required +@handle_exception +def delete_invoice_route(invoice_id): + delete_invoice_data(invoice_id, current_user.id) + log_action("Delete invoice", f"deleted invoice '{invoice_id}'") + return jsonify({"status": "success", "message": f"Invoice {invoice_id} deleted successfully."}) \ No newline at end of file diff --git a/controllers/log_controller.py b/controllers/log_controller.py new file mode 100644 index 0000000..ab6bd30 --- /dev/null +++ b/controllers/log_controller.py @@ -0,0 +1,30 @@ +from flask import Blueprint, render_template, request +from flask_login import login_required + +from model.Log import LogData + +log_bp = Blueprint('log', __name__) + + +@log_bp.route('/activity_log', methods=['GET', 'POST']) +@login_required +def activity_log(): + + start_date = request.values.get("start_date") + end_date = request.values.get("end_date") + user_name = request.values.get("username") + + logData = LogData() + + filtered_logs = logData.GetFilteredActivitiesLog( + start_date, + end_date, + user_name + ) + return render_template( + "activity_log.html", + logs=filtered_logs, + start_date=start_date, + end_date=end_date, + username=user_name + ) \ No newline at end of file diff --git a/controllers/payment_controller.py b/controllers/payment_controller.py new file mode 100644 index 0000000..3973f36 --- /dev/null +++ b/controllers/payment_controller.py @@ -0,0 +1,101 @@ +from flask import Blueprint, render_template, request, redirect, url_for, jsonify, flash +from flask_login import login_required, current_user +from model.payment import Paymentmodel +from model.Log import LogHelper + +payment_bp = Blueprint('payment_bp', __name__) + +# ------------------- Add Payment ------------------- +@payment_bp.route('/add_payment', methods=['GET', 'POST']) +@login_required +def add_payment(): + payments_dicts = Paymentmodel.fetch_all_payments() + # Convert to array for template + payments = [ + [ + p['Payment_Id'], p['PMC_No'], p['Invoice_No'], + p['Payment_Amount'], p['TDS_Payment_Amount'], p['Total_Amount'], p['UTR'] + ] for p in payments_dicts + ] if payments_dicts else [] + + if request.method == 'POST': + subcontractor_id = request.form.get('subcontractor_id') + pmc_no = request.form['PMC_No'] + invoice_no = request.form['invoice_No'] + amount = request.form['Payment_Amount'] + tds_amount = request.form['TDS_Payment_Amount'] + total_amount = request.form['total_amount'] + utr = request.form['utr'] + + LogHelper.log_action("Add Payment", f"User {current_user.id} Add Payment '{pmc_no}'") + Paymentmodel.insert_payment(subcontractor_id,pmc_no, invoice_no, amount, tds_amount, total_amount, utr) + # Paymentmodel.update_inpayment(subcontractor_id, pmc_no, invoice_no, amount, tds_amount, total_amount, utr) + + return redirect(url_for('payment_bp.add_payment')) + + return render_template('add_payment.html', payments=payments) + + +# ------------------- Get PMC Nos ------------------- +@payment_bp.route('/get_pmc_nos_by_subcontractor/') +@login_required +def get_pmc_nos_by_subcontractor(subcontractorId): + connection = Paymentmodel.get_connection() + cur = connection.cursor() + cur.callproc('GetDistinctPMCNoByContractorId', [subcontractorId]) + results = [] + for result in cur.stored_results(): + results = result.fetchall() + cur.close() + pmc_nos = [row[0] for row in results] + return jsonify({'pmc_nos': pmc_nos}) + + +# ------------------- Edit Payment ------------------- +@payment_bp.route('/edit_payment/', methods=['GET', 'POST']) +@login_required +def edit_payment(payment_id): + payment_data = Paymentmodel.fetch_payment_by_id(payment_id) + + if not payment_data: + return "Payment not found", 404 + + if request.method == 'POST': + pmc_no = request.form['PMC_No'] + invoice_no = request.form['invoice_No'] + amount = request.form['Payment_Amount'] + tds_amount = request.form['TDS_Payment_Amount'] + total_amount = request.form['total_amount'] + utr = request.form['utr'] + + LogHelper.log_action("Edit Payment", f"User {current_user.id} Edit Payment '{pmc_no}'") + Paymentmodel.call_update_payment_proc(payment_id, pmc_no, invoice_no, amount, tds_amount, total_amount, utr) + + # Update inpayment + connection = Paymentmodel.get_connection() + cursor = connection.cursor() + cursor.callproc( + 'UpdateInpaymentByPMCInvoiceUTR', + [amount, tds_amount, total_amount, pmc_no, invoice_no, utr] +) + connection.commit() + cursor.close() + connection.close() + + return redirect(url_for('payment_bp.add_payment')) + + return render_template('edit_payment.html', payment_data=payment_data) + +# ------------------- Delete Payment ------------------- +@payment_bp.route('/delete_payment/', methods=['GET']) +@login_required +def delete_payment(payment_id): + success, pmc_no, invoice_no = Paymentmodel.delete_payment(payment_id) + + if not success: + flash("Payment not found or failed to delete", "error") + else: + LogHelper.log_action("Delete Payment", f"User {current_user.id} deleted Payment '{payment_id}'") + flash(f"Payment ID {payment_id} deleted successfully.", "success") + + return redirect(url_for('payment_bp.add_payment')) \ No newline at end of file diff --git a/controllers/pmc_report_controller.py b/controllers/pmc_report_controller.py new file mode 100644 index 0000000..76460e7 --- /dev/null +++ b/controllers/pmc_report_controller.py @@ -0,0 +1,40 @@ +from flask import Blueprint, render_template, send_from_directory +from flask_login import login_required + +from model.PmcReport import PmcReport + +pmc_report_bp = Blueprint("pmc_report", __name__) + +# ---------------- Contractor Report by pmc no ---------------- +@pmc_report_bp.route("/pmc_report/") +@login_required +def pmc_report(pmc_no): + data = PmcReport.get_pmc_report(pmc_no) + if not data: + return "No PMC found with this number", 404 + + return render_template( + "pmc_report.html", + info=data["info"], + invoices=data["invoices"], + hold_types=data["hold_types"], + gst_rel=data["gst_rel"], + payments=data["payments"], + credit_note=data["credit_note"], + hold_release=data["hold_release"], + total=data["total"] + ) + +# ---------------- Contractor Download Report by pmc no ---------------- +@pmc_report_bp.route("/download_pmc_report/") +@login_required +def download_pmc_report(pmc_no): + + result = PmcReport.download_pmc_report(pmc_no) + + if not result: + return "No contractor found for this PMC No", 404 + + output_folder, file_name = result + + return send_from_directory(output_folder, file_name, as_attachment=True) \ No newline at end of file diff --git a/controllers/report_controller.py b/controllers/report_controller.py new file mode 100644 index 0000000..e8d6d15 --- /dev/null +++ b/controllers/report_controller.py @@ -0,0 +1,67 @@ +from flask import Blueprint, render_template, request, jsonify,send_file +from flask_login import login_required, current_user + +from model.Report import ReportHelper +from model.Log import LogHelper +import os + +DOWNLOAD_FOLDER = "static/download" +report_bp = Blueprint("report", __name__) + + +# ---------------- Report Page ---------------- +@report_bp.route("/report") +@login_required +def report_page(): + return render_template("/report.html") + + +# ---------------- Search Contractor ---------------- +@report_bp.route("/search_contractor", methods=["POST"]) +@login_required +def search_contractor(): + + subcontractor_name = request.form.get("subcontractor_name") + + LogHelper.log_action( + "Search Contractor", + f"User {current_user.id} searched contractor '{subcontractor_name}'" + ) + + data = ReportHelper.search_contractor(request) + + return jsonify(data) + +# ---------------- Contractor Report by contractor id ---------------- +@report_bp.route('/contractor_report/') +@login_required +def contractor_report(contractor_id): + + data = ReportHelper.get_contractor_report(contractor_id) + + return render_template( + 'subcontractor_report.html', + contractor_id=contractor_id, + **data + ) + +# # ---------------- Contractor Download Report by contractor id ---------------- +# @report_bp.route('/download_report/') +# @login_required +# def download_report(contractor_id): + +# return ReportHelper().download_report(contractor_id=contractor_id) + + + + + +@report_bp.route('/download_report/') +def download_report(contractor_id): + output_file, error = ReportHelper.create_contractor_report(contractor_id) + + if error: + return error, 404 + + # Send the file to the user + return send_file(output_file, as_attachment=True) \ No newline at end of file diff --git a/controllers/state_controller.py b/controllers/state_controller.py new file mode 100644 index 0000000..a26c951 --- /dev/null +++ b/controllers/state_controller.py @@ -0,0 +1,69 @@ +from flask import Blueprint, render_template, request, redirect, url_for +from flask_login import login_required +from model.State import State + +state_bp = Blueprint('state', __name__) + +# ----- State page ------ +@state_bp.route('/add_state', methods=['GET', 'POST']) +@login_required +def add_state(): + + state = State() + + if request.method == 'POST': + state.AddState(request=request) + return state.resultMessage + + statedata = state.GetAllStates(request=request) + + return render_template('add_state.html', statedata=statedata) + +# ----- State check ------ +@state_bp.route('/check_state', methods=['POST']) +@login_required +def check_state(): + + state = State() + + return state.CheckState(request=request) + +# ----- State delete by state id ------ +@state_bp.route('/delete_state/') +@login_required +def deleteState(id): + + state = State() + + state.DeleteState(request=request, id=id) + + if not state.isSuccess: + return state.resultMessage + else: + return redirect(url_for('state.add_state')) + +# ----- State update by state id ------ +@state_bp.route('/edit_state/', methods=['GET', 'POST']) +@login_required +def editState(id): + + state = State() + + if request.method == 'POST': + + state.EditState(request=request, id=id) + + if state.isSuccess: + return redirect(url_for('state.add_state')) + else: + return state.resultMessage + + statedata = state.GetStateByID(request=request, id=id) + + if not state.isSuccess: + return state.resultMessage + + if statedata is None: + statedata = [] + + return render_template('edit_state.html', state=statedata) \ No newline at end of file diff --git a/controllers/subcontractor_controller.py b/controllers/subcontractor_controller.py new file mode 100644 index 0000000..d9f7fbd --- /dev/null +++ b/controllers/subcontractor_controller.py @@ -0,0 +1,94 @@ +from flask import Blueprint, render_template, request, redirect, url_for, jsonify +from flask_login import login_required +from model.Subcontractor import Subcontractor +from model.Utilities import HtmlHelper, ResponseHandler + + +subcontractor_bp = Blueprint('subcontractor', __name__) + + +# ---------------------------------------------------------- +# LIST + ADD +# ---------------------------------------------------------- +@subcontractor_bp.route('/subcontractor', methods=['GET', 'POST']) +@login_required +def subcontract(): + + sub = Subcontractor() + + # ---------------- GET ---------------- + if request.method == 'GET': + subcontractor = sub.GetAllSubcontractors(request) + + if not sub.isSuccess: + return HtmlHelper.json_response( + ResponseHandler.fetch_failure("Subcontractor"), 500 + ) + + return render_template('add_subcontractor.html', subcontractor=subcontractor) + + # ---------------- POST (ADD) ---------------- + if request.method == 'POST': + + sub.AddSubcontractor(request) + + if not sub.isSuccess: + return HtmlHelper.json_response( + ResponseHandler.add_failure("Subcontractor"), 500 + ) + + # Reload list after insert + subcontractor = sub.GetAllSubcontractors(request) + + return render_template('add_subcontractor.html', subcontractor=subcontractor) + + +# ---------------------------------------------------------- +# EDIT +# ---------------------------------------------------------- +@subcontractor_bp.route('/edit_subcontractor/', methods=['GET', 'POST']) +@login_required +def edit_subcontractor(id): + + sub = Subcontractor() + + # Fetch data + subcontractor = sub.GetSubcontractorByID(id) + + if not subcontractor: + return HtmlHelper.json_response( + ResponseHandler.fetch_failure("Subcontractor"), 404 + ) + + # ---------------- POST (UPDATE) ---------------- + if request.method == 'POST': + + sub.EditSubcontractor(request, id) + + if not sub.isSuccess: + return HtmlHelper.json_response( + ResponseHandler.update_failure("Subcontractor"), 500 + ) + + return redirect(url_for('subcontractor.subcontract')) + + return render_template('edit_subcontractor.html', subcontractor=subcontractor) + + +# ---------------------------------------------------------- +# DELETE +# ---------------------------------------------------------- +@subcontractor_bp.route('/deleteSubContractor/', methods=['GET', 'POST']) +@login_required +def deleteSubContractor(id): + + sub = Subcontractor() + + sub.DeleteSubcontractor(request, id) + + if not sub.isSuccess: + return HtmlHelper.json_response( + ResponseHandler.delete_failure("Subcontractor"), 500 + ) + + return redirect(url_for('subcontractor.subcontract')) \ No newline at end of file diff --git a/controllers/village_controller.py b/controllers/village_controller.py new file mode 100644 index 0000000..27a1ff2 --- /dev/null +++ b/controllers/village_controller.py @@ -0,0 +1,151 @@ + + + + +from flask import Blueprint, render_template, request, redirect, url_for, flash, jsonify +from flask_login import login_required + +import config +from model.Village import Village +from model.State import State + +# Create Blueprint +village_bp = Blueprint('village', __name__) + + +# ------------------------- Add Village ------------------------- +@village_bp.route('/add_village', methods=['GET', 'POST']) +@login_required +def add_village(): + + village = Village() + + if request.method == 'POST': + village.AddVillage(request=request) + return village.resultMessage + + state = State() + states = state.GetAllStates(request=request) + villages = village.GetAllVillages(request=request) + + return render_template( + 'add_village.html', + states=states, + villages=villages + ) + + +# ------------------------- Fetch Districts ------------------------- +@village_bp.route('/get_districts/') +@login_required +def get_districts(state_id): + + connection = config.get_db_connection() + cursor = connection.cursor() + + cursor.callproc("GetDistrictByStateID", [state_id]) + districts = [] + + for rs in cursor.stored_results(): + districts = rs.fetchall() + + cursor.close() + connection.close() + + return jsonify([{"id": d[0], "name": d[1]} for d in districts]) + + +# ------------------------- Fetch Blocks ------------------------- +@village_bp.route('/get_blocks/') +@login_required +def get_blocks(district_id): + + connection = config.get_db_connection() + cursor = connection.cursor() + + cursor.callproc("GetBlocksByDistrictID", [district_id]) + blocks = [] + + for rs in cursor.stored_results(): + blocks = rs.fetchall() + + cursor.close() + connection.close() + + return jsonify([{"id": b[0], "name": b[1]} for b in blocks]) + + +# ------------------------- Check Village ------------------------- +@village_bp.route('/check_village', methods=['POST']) +@login_required +def check_village(): + village = Village() + return village.CheckVillage(request=request) + + +@village_bp.route('/delete_village/') +@login_required +def delete_village(village_id): + village = Village() + village.DeleteVillage(request=request, village_id=village_id) + + # ✅ Convert resultMessage to string if it's a Response or tuple + raw_msg = village.resultMessage + + if isinstance(raw_msg, tuple): + # e.g., (, 200) + msg = "Village deleted successfully!" + elif hasattr(raw_msg, 'get_data'): + # Flask Response object + msg = raw_msg.get_data(as_text=True) # get raw text + else: + # fallback + msg = str(raw_msg) if raw_msg else "Village deleted successfully!" + + return jsonify({ + "status": "success" if village.isSuccess else "error", + "message": msg + }) + +# ------------------------- Edit Village ------------------------- +@village_bp.route('/edit_village/', methods=['GET', 'POST']) +@login_required +def edit_village(village_id): + + village = Village() + + if request.method == 'POST': + + village.EditVillage(request=request, village_id=village_id) + + if village.isSuccess: + flash(village.resultMessage, "success") + return redirect(url_for('village.add_village')) + + else: + flash(village.resultMessage, "error") + + village_data = village.GetVillageByID(id=village_id) or [] + blocks = village.GetAllBlocks() or [] + + return render_template( + 'edit_village.html', + village_data=village_data, + blocks=blocks + ) + + else: + # ✅ FIXED HERE (removed request) + village_data = village.GetVillageByID(id=village_id) + + if not village.isSuccess: + flash(village.resultMessage, "error") + return redirect(url_for('village.add_village')) + + blocks = village.GetAllBlocks() or [] + + return render_template( + 'edit_village.html', + village_data=village_data or [], + blocks=blocks + ) \ No newline at end of file diff --git a/docker-compose.yml b/docker-compose.yml new file mode 100644 index 0000000..2a3c49d --- /dev/null +++ b/docker-compose.yml @@ -0,0 +1,35 @@ +version: "3.8" + +services: + flask-app: + build: . + ports: + - "5000:5000" + depends_on: + - mysql + environment: + - MYSQL_HOST=mysql + - MYSQL_USER=root + - MYSQL_PASSWORD=root + - MYSQL_DB=test + networks: + - mynetwork + + mysql: + image: mysql:8.0 + restart: always + environment: + MYSQL_ROOT_PASSWORD: root + MYSQL_DATABASE: test + ports: + - "3306:3306" + volumes: + - mysql-data:/var/lib/mysql + networks: + - mynetwork + +volumes: + mysql-data: + +networks: + mynetwork: diff --git a/download/Contractor_Report_1.xlsx b/download/Contractor_Report_1.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..6b4e1ded4d3fb267b06144c86e9af7fada0ff0c5 GIT binary patch literal 8218 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 literal 0 HcmV?d00001 diff --git a/logs/activity.log b/logs/activity.log new file mode 100644 index 0000000..d00d466 --- /dev/null +++ b/logs/activity.log @@ -0,0 +1,2405 @@ +Timestamp: 2026-03-25 15:21:23 | User: Unknown | Action: Login | Details: +Timestamp: 2026-03-25 15:21:29 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 15:21:31 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 15:21:31 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 15:22:08 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-25 15:23:07 | User: Unknown | Action: Logout | Details: +Timestamp: 2026-03-25 15:23:12 | User: Unknown | Action: Login | Details: +Timestamp: 2026-03-25 15:23:45 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-25 15:24:27 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-25 15:24:31 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 15:24:45 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 15:24:47 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 15:29:41 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 15:29:44 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 16:28:43 | User: Unknown | Action: Login | Details: +Timestamp: 2026-03-25 17:03:45 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-25 17:04:27 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-25 17:04:55 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-25 17:04:59 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:05:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:17:33 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-25 17:17:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:18:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:17 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:19:21 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-25 17:38:51 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 17:38:52 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 17:38:53 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 19:48:55 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 19:48:55 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 19:48:57 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 20:44:16 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-25 20:44:18 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 13:42:24 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 13:42:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 13:42:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 14:23:46 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 14:24:04 | User: Unknown | Action: Add Subcontractor | Details: +Timestamp: 2026-03-26 14:24:23 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 14:25:09 | User: Unknown | Action: Add invoice | Details: +Timestamp: 2026-03-26 14:25:11 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 14:26:54 | User: Unknown | Action: Edit invoice | Details: +Timestamp: 2026-03-26 14:26:56 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 14:27:03 | User: Unknown | Action: Delete invoice | Details: +Timestamp: 2026-03-26 14:40:54 | User: Unknown | Action: Add invoice | Details: +Timestamp: 2026-03-26 14:40:56 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 14:41:07 | User: Unknown | Action: Edit invoice | Details: +Timestamp: 2026-03-26 14:41:09 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 14:41:46 | User: Unknown | Action: Add Payment | Details: +Timestamp: 2026-03-26 14:42:03 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 14:42:06 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 14:50:46 | User: Unknown | Action: Add GSTRelease | Details: +Timestamp: 2026-03-26 14:50:46 | User: Unknown | Action: Add GST Release | Details: +Timestamp: 2026-03-26 14:51:07 | User: Unknown | Action: Add GSTRelease | Details: +Timestamp: 2026-03-26 14:51:07 | User: Unknown | Action: Add GST Release | Details: +Timestamp: 2026-03-26 14:53:56 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 14:53:58 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:15:28 | User: Unknown | Action: Login | Details: +Timestamp: 2026-03-26 17:15:30 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 17:15:38 | User: Unknown | Action: Delete invoice | Details: +Timestamp: 2026-03-26 17:15:54 | User: Unknown | Action: Delete Payment | Details: +Timestamp: 2026-03-26 17:16:00 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 17:16:24 | User: Unknown | Action: Add invoice | Details: +Timestamp: 2026-03-26 17:16:26 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 17:19:29 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 17:19:51 | User: Unknown | Action: Add invoice | Details: +Timestamp: 2026-03-26 17:19:53 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 17:20:21 | User: Unknown | Action: Add Payment | Details: +Timestamp: 2026-03-26 17:22:41 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 17:22:49 | User: Unknown | Action: Delete invoice | Details: +Timestamp: 2026-03-26 17:23:17 | User: Unknown | Action: Add Payment | Details: +Timestamp: 2026-03-26 17:24:37 | User: Unknown | Action: Add Payment | Details: +Timestamp: 2026-03-26 17:24:48 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:29:15 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 17:29:37 | User: Unknown | Action: Add invoice | Details: +Timestamp: 2026-03-26 17:29:39 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 17:30:09 | User: Unknown | Action: Add Payment | Details: +Timestamp: 2026-03-26 17:30:16 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:30:18 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:31:21 | User: Unknown | Action: Add Payment | Details: +Timestamp: 2026-03-26 17:31:31 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:31:33 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:36:00 | User: Unknown | Action: Add Payment | Details: +Timestamp: 2026-03-26 17:40:30 | User: Unknown | Action: Add Payment | Details: +Timestamp: 2026-03-26 17:42:45 | User: Unknown | Action: Add Payment | Details: +Timestamp: 2026-03-26 17:42:58 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:43:00 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:51:13 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 17:51:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:51:45 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 17:51:47 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 17:52:11 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 17:52:18 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:19 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:28 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:52:53 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:52:55 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:52:56 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:52:56 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:52:56 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:52:57 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:52:57 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:52:57 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:52:58 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:53:00 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:57:34 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 17:57:50 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 17:57:53 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 18:49:19 | User: Unknown | Action: Login | Details: +Timestamp: 2026-03-26 18:49:23 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 18:49:23 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 18:49:25 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 19:18:26 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 19:18:41 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 19:18:42 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 19:58:36 | User: Unknown | Action: Add Subcontractor | Details: +Timestamp: 2026-03-26 19:58:42 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 19:59:13 | User: Unknown | Action: Add invoice | Details: +Timestamp: 2026-03-26 19:59:15 | User: Unknown | Action: Get hold type | Details: +Timestamp: 2026-03-26 19:59:47 | User: Unknown | Action: Add Payment | Details: +Timestamp: 2026-03-26 20:00:22 | User: Unknown | Action: Add GSTRelease | Details: +Timestamp: 2026-03-26 20:00:22 | User: Unknown | Action: Add GST Release | Details: +Timestamp: 2026-03-26 20:00:34 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 20:00:35 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 20:01:06 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 20:01:07 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 20:25:02 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 20:25:05 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:25:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:25:26 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 20:25:27 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 20:28:30 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:28:46 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 20:28:48 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 20:33:08 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:33:11 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:00 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:38:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:49:34 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 20:49:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:49:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:49:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:49:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:49:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:49:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:49:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:49:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:49:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:49:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:49:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:50:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:50:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:50:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:50:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:50:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:50:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:50:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:50:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:50:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:50:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:50:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:46 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 20:59:59 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 21:00:01 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 21:08:24 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 21:08:28 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:08:28 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:08:28 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:08:28 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:10:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:10:15 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 21:10:17 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 21:14:31 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 21:14:34 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:14:34 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:14:34 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:14:34 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:15:37 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 21:15:40 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:15:41 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:15:41 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:15:41 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:21:50 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 21:21:53 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:21:53 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:21:53 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:21:54 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:02 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-26 21:25:06 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:06 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:07 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-26 21:25:18 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-26 21:25:20 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 10:01:46 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:01:49 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:09:33 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 10:09:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:09:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:41 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:12:44 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:13:24 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 10:13:26 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 10:17:12 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:17:26 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 10:17:26 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 10:17:27 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 10:34:04 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 10:50:19 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 10:50:22 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:23 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:50:29 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 10:50:30 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 10:51:31 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:35 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:51:45 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 10:51:48 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 10:53:39 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:47 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 10:53:48 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:05 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:13 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:14 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:15 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:16 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:17 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:00:28 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 11:00:29 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 11:27:26 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:29 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:30 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 11:27:40 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 11:27:40 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 11:27:42 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 11:40:20 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 11:40:20 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 11:40:20 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 11:40:22 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 11:40:22 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 11:40:22 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:18:36 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 12:18:39 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:18:48 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:18:48 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:18:50 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:22:57 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:01 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:23:14 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:23:15 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:23:16 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:33:43 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:46 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:33:55 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:33:58 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:49:48 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:49:48 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:49:49 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:49:51 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:54:15 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:54:15 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:54:17 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:54:19 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:54:19 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:54:19 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:54:34 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 12:54:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:54:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:55:14 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:55:14 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:55:16 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:58:25 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:36 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:37 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:58:38 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 12:59:44 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:59:45 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:59:45 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 12:59:46 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 13:05:45 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 13:05:55 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:55 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:55 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:55 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:55 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:55 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:55 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:55 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:56 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:05:57 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:06:07 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 13:06:08 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 13:06:09 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 13:13:46 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:13:50 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:18:00 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 13:18:03 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:18:36 | User: Unknown | Action: Upload Excel File | Details: +Timestamp: 2026-03-27 13:18:40 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:18:40 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:18:40 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:18:40 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:18:40 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:18:40 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:18:40 | User: Unknown | Action: Data saved | Details: +Timestamp: 2026-03-27 13:18:54 | User: Unknown | Action: Search Contractor | Details: +Timestamp: 2026-03-27 13:18:56 | User: Unknown | Action: Search Contractor | Details: diff --git a/logs/audit.log b/logs/audit.log new file mode 100644 index 0000000..02c0a11 --- /dev/null +++ b/logs/audit.log @@ -0,0 +1,74 @@ +2025-08-22 13:29:24,485 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.180 +2025-08-22 13:41:42,046 | User: v.sinha | Action: Logout | Details: User v.sinha logged out | IP: 192.168.0.180 +2025-08-22 14:07:01,924 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.180 +2025-08-22 15:05:59,287 | User: v.sinha | Action: Logout | Details: User v.sinha logged out | IP: 192.168.0.180 +2025-08-22 15:06:05,201 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.180 +2025-08-23 15:58:28,248 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.180 +2025-08-23 17:33:06,648 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.180 +2025-08-23 17:39:08,442 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.180 +2025-08-23 18:14:51,722 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.180 +2025-08-25 11:57:12,202 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.238 +2025-08-25 12:00:17,780 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.181 +2025-08-25 14:09:29,385 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.181 +2025-08-25 14:12:35,084 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.181 +2025-08-25 14:23:53,539 | User: v.sinha | Action: Checked State | Details: User MP Checking State | IP: 192.168.0.181 +2025-08-25 14:23:54,024 | User: v.sinha | Action: Added State | Details: User MP Adding State | IP: 192.168.0.181 +2025-08-25 14:23:57,113 | User: v.sinha | Action: Deleted State | Details: User 11 Deleting State | IP: 192.168.0.181 +2025-08-25 14:31:55,715 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.181 +2025-08-25 14:36:21,158 | User: v.sinha | Action: Checked State | Details: User MP Checking State | IP: 192.168.0.181 +2025-08-25 14:36:21,496 | User: v.sinha | Action: Added State | Details: State 'MP' added. | IP: 192.168.0.181 +2025-08-25 14:47:08,719 | User: v.sinha | Action: Checked State | Details: User Maharashtra Checking State | IP: 192.168.0.181 +2025-08-25 14:47:14,759 | User: v.sinha | Action: Deleted State | Details: User 12 Deleting State | IP: 192.168.0.181 +2025-08-25 14:47:16,915 | User: v.sinha | Action: Checked State | Details: User MP Checking State | IP: 192.168.0.181 +2025-08-25 14:47:17,708 | User: v.sinha | Action: Added State | Details: State 'MP' added. | IP: 192.168.0.181 +2025-08-25 14:49:09,480 | User: v.sinha | Action: Checked State | Details: User MP Checking State | IP: 192.168.0.181 +2025-08-25 14:49:13,014 | User: v.sinha | Action: Deleted State | Details: User 13 Deleting State | IP: 192.168.0.181 +2025-08-25 14:49:14,584 | User: v.sinha | Action: Checked State | Details: User MP Checking State | IP: 192.168.0.181 +2025-08-25 14:49:15,055 | User: v.sinha | Action: Added State | Details: State 'MP' added. | IP: 192.168.0.181 +2025-08-25 14:51:55,187 | User: v.sinha | Action: Checked State | Details: User MP Checking State | IP: 192.168.0.181 +2025-08-25 14:51:58,463 | User: v.sinha | Action: Deleted State | Details: User 14 Deleting State | IP: 192.168.0.181 +2025-08-25 14:52:00,606 | User: v.sinha | Action: Checked State | Details: User MP Checking State | IP: 192.168.0.181 +2025-08-25 14:52:00,953 | User: v.sinha | Action: Added State | Details: State 'MP' added. | IP: 192.168.0.181 +2025-08-25 14:54:26,674 | User: v.sinha | Action: Deleted State | Details: User 15 Deleting State | IP: 192.168.0.181 +2025-08-25 14:54:28,892 | User: v.sinha | Action: Checked State | Details: User MP Checking State | IP: 192.168.0.181 +2025-08-25 14:54:29,553 | User: v.sinha | Action: Added State | Details: State 'MP' added. | IP: 192.168.0.181 +2025-08-25 15:18:37,773 | User: v.sinha | Action: Logout | Details: User v.sinha logged out | IP: 192.168.0.181 +2025-08-25 15:18:43,347 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.181 +2025-08-25 15:20:41,331 | User: v.sinha | Action: Logout | Details: User v.sinha logged out | IP: 192.168.0.181 +2025-08-25 15:20:47,525 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.181 +2025-08-25 15:20:55,687 | User: v.sinha | Action: Checked State | Details: User MP Checking State | IP: 192.168.0.181 +2025-08-25 15:20:58,544 | User: v.sinha | Action: Deleted State | Details: User 16 Deleting State | IP: 192.168.0.181 +2025-08-25 16:33:49,898 | User: v.sinha | Action: Check State | Details: User v.sinha Checked state 'MP' | IP: 192.168.0.181 +2025-08-25 16:33:50,394 | User: v.sinha | Action: Add State | Details: User v.sinha added state 'MP' | IP: 192.168.0.181 +2025-08-25 16:43:46,446 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.181 +2025-08-25 16:43:49,710 | User: v.sinha | Action: Logout | Details: User v.sinha logged out | IP: 192.168.0.181 +2025-08-25 16:43:58,093 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.181 +2025-08-25 16:44:11,935 | User: v.sinha | Action: Check State | Details: User v.sinha Checked state 'shamli' | IP: 192.168.0.181 +2025-08-25 16:44:12,466 | User: v.sinha | Action: Add State | Details: User v.sinha added state 'shamli' | IP: 192.168.0.181 +2025-08-25 16:44:17,731 | User: v.sinha | Action: Delete State | Details: User v.sinha Deleted state '18' | IP: 192.168.0.181 +2025-08-25 16:57:27,983 | User: v.sinha | Action: Logout | Details: User v.sinha logged out | IP: 192.168.0.181 +2025-08-25 16:57:33,498 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.181 +2025-08-25 16:57:41,438 | User: v.sinha | Action: Check State | Details: User v.sinha Checked state 'shamli' | IP: 192.168.0.181 +2025-08-25 16:57:42,250 | User: v.sinha | Action: Add State | Details: User v.sinha added state 'shamli' | IP: 192.168.0.181 +2025-08-25 16:57:45,339 | User: v.sinha | Action: Delete State | Details: User v.sinha Deleted state '19' | IP: 192.168.0.181 +2025-08-25 16:57:48,794 | User: v.sinha | Action: Delete State | Details: User v.sinha Deleted state '17' | IP: 192.168.0.181 +2025-08-25 17:04:11,021 | User: v.sinha | Action: Logout | Details: User v.sinha logged out | IP: 192.168.0.181 +2025-08-25 17:04:16,165 | User: v.sinha | Action: Login | Details: User v.sinha logged in | IP: 192.168.0.181 +2025-08-25 17:04:21,702 | User: v.sinha | Action: Check State | Details: User v.sinha Checked state 'shamli' | IP: 192.168.0.181 +2025-08-25 17:04:22,159 | User: v.sinha | Action: Add State | Details: User v.sinha added state 'shamli' | IP: 192.168.0.181 +2025-08-25 17:04:26,850 | User: v.sinha | Action: Check State | Details: User v.sinha Checked state 'M' | IP: 192.168.0.181 +2025-08-25 17:04:27,076 | User: v.sinha | Action: Check State | Details: User v.sinha Checked state 'MP' | IP: 192.168.0.181 +2025-08-25 17:04:28,070 | User: v.sinha | Action: Add State | Details: User v.sinha added state 'MP' | IP: 192.168.0.181 +2025-08-25 17:04:32,165 | User: v.sinha | Action: Delete State | Details: User v.sinha Deleted state '21' | IP: 192.168.0.181 +2025-08-25 17:04:35,058 | User: v.sinha | Action: Delete State | Details: User v.sinha Deleted state '20' | IP: 192.168.0.181 +2025-08-25 17:06:05,113 | User: v.sinha | Action: Check District | Details: User v.sinha Checked District 'Shamli' | IP: 192.168.0.181 +2025-08-25 17:06:05,114 | User: v.sinha | Action: Check District | Details: User v.sinha Checked District 'Shamli' | IP: 192.168.0.181 +2025-08-25 17:06:08,040 | User: v.sinha | Action: Check District | Details: User v.sinha Checked District 'p' | IP: 192.168.0.181 +2025-08-25 17:06:08,360 | User: v.sinha | Action: Check District | Details: User v.sinha Checked District 'pu' | IP: 192.168.0.181 +2025-08-25 17:06:08,554 | User: v.sinha | Action: Check District | Details: User v.sinha Checked District 'pun' | IP: 192.168.0.181 +2025-08-25 17:06:08,756 | User: v.sinha | Action: Check District | Details: User v.sinha Checked District 'pune' | IP: 192.168.0.181 +2025-08-25 17:06:10,190 | User: v.sinha | Action: Check District | Details: User v.sinha Checked District 'pune' | IP: 192.168.0.181 +2025-08-25 17:06:11,204 | User: v.sinha | Action: Check District | Details: User v.sinha Checked District 'pune' | IP: 192.168.0.181 +2025-08-25 17:06:11,206 | User: v.sinha | Action: Check District | Details: User v.sinha Checked District 'pune' | IP: 192.168.0.181 +2025-08-25 17:06:13,085 | User: v.sinha | Action: Add District | Details: User v.sinha Added District 'pune' | IP: 192.168.0.181 +2025-08-25 17:06:19,524 | User: v.sinha | Action: Delete District | Details: User v.sinha Deleted District '5' | IP: 192.168.0.181 diff --git a/main.py b/main.py new file mode 100644 index 0000000..96a2977 --- /dev/null +++ b/main.py @@ -0,0 +1,68 @@ +# main.py +from flask import Flask, render_template +from flask_login import LoginManager +from model.Auth import User + +# Import Blueprints / Controllers +from controllers.auth_controller import auth_bp +from controllers.log_controller import log_bp +from controllers.state_controller import state_bp +from controllers.district_controller import district_bp +from controllers.block_controller import block_bp +from controllers.village_controller import village_bp +from controllers.invoice_controller import invoice_bp +from controllers.subcontractor_controller import subcontractor_bp +from controllers.payment_controller import payment_bp +from controllers.gst_release_controller import gst_release_bp +from controllers.excel_upload_controller import excel_bp +from controllers.report_controller import report_bp +from controllers.pmc_report_controller import pmc_report_bp + +from controllers.hold_types_controller import hold_bp +from dotenv import load_dotenv +import os + +# ---------------- Initialize App ---------------- +app = Flask(__name__) + +load_dotenv() +Secret_Key = os.getenv("SECRET_KEY") +app.secret_key = Secret_Key + +# ---------------- Login Manager ---------------- +login_manager = LoginManager() +login_manager.init_app(app) +login_manager.login_view = 'auth.login' + +@login_manager.user_loader +def load_user(user_id): + return User(user_id) + +# ---------------- Home Route ---------------- +@app.route('/') +def index(): + return render_template("index.html") + +# ---------------- Register Blueprints ---------------- + +app.register_blueprint(auth_bp) +app.register_blueprint(log_bp) +app.register_blueprint(state_bp) +app.register_blueprint(district_bp) +app.register_blueprint(block_bp) +app.register_blueprint(village_bp) +app.register_blueprint(invoice_bp) +app.register_blueprint(subcontractor_bp) +app.register_blueprint(payment_bp) +app.register_blueprint(gst_release_bp) +app.register_blueprint(excel_bp) +app.register_blueprint(report_bp) +app.register_blueprint(pmc_report_bp) +app.register_blueprint(hold_bp) + +# ---------------- Run App ---------------- +if __name__ == '__main__': + app.run(host='0.0.0.0', port=5000, debug=True) + + + diff --git a/model/Auth.py b/model/Auth.py new file mode 100644 index 0000000..e1b9cd4 --- /dev/null +++ b/model/Auth.py @@ -0,0 +1,63 @@ +import os +from dotenv import load_dotenv +from flask_login import UserMixin +from ldap3 import Server, Connection, ALL +from ldap3.core.exceptions import LDAPBindError + +# Load .env +load_dotenv() + + +class DefaultCredentials: + username = os.getenv("DEFAULT_USERNAME") + password = os.getenv("DEFAULT_PASSWORD") + + +class LoginLDAP: + + def __init__(self, request): + + self.username = request.form.get("username", "").strip() + self.password = request.form.get("password", "") + + self.isDefaultCredentials = False + self.isValidLogin = False + self.errorMessage = "" + + ldap_server = "ldap://localhost:389" + ldap_user_dn = f"uid={self.username},ou=users,dc=lcepl,dc=org" + + # fallback admin login + if ( + self.username == DefaultCredentials.username + and self.password == DefaultCredentials.password + ): + self.isDefaultCredentials = True + self.isValidLogin = True + return + + try: + + server = Server(ldap_server, get_info=ALL) + + conn = Connection( + server, + user=ldap_user_dn, + password=self.password, + auto_bind=True + ) + + if conn.bound: + self.isValidLogin = True + + except LDAPBindError: + self.errorMessage = "Invalid LDAP credentials" + + except Exception as e: + self.errorMessage = str(e) + + +class User(UserMixin): + + def __init__(self, username): + self.id = username \ No newline at end of file diff --git a/model/Block.py b/model/Block.py new file mode 100644 index 0000000..640ab0c --- /dev/null +++ b/model/Block.py @@ -0,0 +1,161 @@ +from flask import Blueprint, render_template, request, redirect, url_for, jsonify + +from model.Utilities import RegEx, ResponseHandler, HtmlHelper, ItemCRUDType +from model.Log import LogData, LogHelper + +import os +import config +import re + +import mysql.connector +from mysql.connector import Error + +from model.ItemCRUD import ItemCRUD, itemCRUDMapping + + +class Block: + + isSuccess = False + resultMessage = "" + + def __init__(self): + self.isSuccess = False + self.resultMessage = "" + + # ---------------------------------------------------------- + # Add Block + # ---------------------------------------------------------- + def AddBlock(self, request): + + block = ItemCRUD(itemType=ItemCRUDType.Block) + + district_id = request.form.get('district_Id') + block_name = request.form.get('block_Name', '').strip() + + block.AddItem(request=request, parentid=district_id, childname=block_name, storedprocfetch="GetBlockByNameAndDistricts", storedprocadd="SaveBlock" ) + self.isSuccess = block.isSuccess + self.resultMessage = block.resultMessage + return + + # ---------------------------------------------------------- + # Get All Blocks + # ---------------------------------------------------------- + # def GetAllBlocks(self): + + # block = ItemCRUD(itemType=ItemCRUDType.Block) + # blocksdata = block.GetAllData(request=request, storedproc="GetAllBlock") + # self.isSuccess = block.isSuccess + # self.resultMessage = block.resultMessage + # return blocksdata + + def GetAllBlocks(self, request): + + block = ItemCRUD(itemType=ItemCRUDType.Block) + blocksdata = block.GetAllData(request=request, storedproc="GetAllBlock") + + self.isSuccess = block.isSuccess + self.resultMessage = block.resultMessage + return blocksdata + + # ---------------------------------------------------------- + # Check Block Exists + # ---------------------------------------------------------- + + # def CheckBlock(self, request): + # block = ItemCRUD(itemType=ItemCRUDType.Block) + # block_name = request.json.get('block_Name', '').strip() + # district_id = request.json.get('district_Id') + # result = block.CheckItem(request=request, parentid=district_id, childname=block_name, storedprocfetch="GetBlockByNameAndDistrict") + # self.isSuccess = block.isSuccess + # self.resultMessage = block.resultMessage + # return result + def CheckBlock(self, request): + block = ItemCRUD(itemType=ItemCRUDType.Block) + data = request.get_json(silent=True) or request.form + block_name = (data.get('block_Name') or '').strip() + district_id = data.get('district_Id') + + result = block.CheckItem( + request=request, + parentid=district_id, + childname=block_name, + storedprocfetch="GetBlockByNameAndDistrict" + ) + self.isSuccess = block.isSuccess + self.resultMessage = block.resultMessage + return result + + # ---------------------------------------------------------- + # Get Block By ID + # ---------------------------------------------------------- + # def GetBlockByID(self, id): + + # block = ItemCRUD(itemType=ItemCRUDType.Village) + # blockdata = block.GetAllData(id=id,storedproc="GetBlockDataByID") + # self.isSuccess = block.isSuccess + # self.resultMessage = block.resultMessage + # print("akash"+blockdata) + # return blockdata + + # def GetBlockByID(self,request,id): + # block = ItemCRUD(itemType=ItemCRUDType.Block) + # blockdata = block.GetDataByID(request=request,id=id,storedproc="GetBlockDataByID") + # self.isSuccess = block.isSuccess + # self.resultMessage = block.resultMessage + # return blockdata + def GetBlockByID(self, id): + + block = ItemCRUD(itemType=ItemCRUDType.Block) + + blockdata = block.GetDataByID( + id=id, + storedproc="GetBlockDataByID" + ) + + self.isSuccess = block.isSuccess + self.resultMessage = block.resultMessage + + return blockdata + # ---------------------------------------------------------- + # Update Block + # ---------------------------------------------------------- + # def EditBlock(self, request, block_id): + + # block = ItemCRUD(itemType=ItemCRUDType.Block) + + # district_id = request.form.get('district_Id') + # block_name = request.form.get('block_Name', '').strip() + + # block.EditItem(request=request, childid=block_id, parentid=district_id, childname=block_name, storedprocadd="UpdateBlockById" ) + # self.isSuccess = block.isSuccess + # self.resultMessage = block.resultMessage + # return + def EditBlock(self, request, block_id): + + block = ItemCRUD(itemType=ItemCRUDType.Block) + + district_id = request.form.get('district_Id') + block_name = request.form.get('block_Name', '').strip() + + block.EditItem( + request=request, + childid=block_id, + parentid=district_id, + childname=block_name, + storedprocupdate="UpdateBlockById" + ) + + self.isSuccess = block.isSuccess + self.resultMessage = block.resultMessage + return + + # ---------------------------------------------------------- + # Delete Block + # --------------------------------------------------------- + def DeleteBlock(self,request, id): + block = ItemCRUD(itemType=ItemCRUDType.Block) + + block.DeleteItem(request=request,itemID=id, storedprocDelete="DeleteBlock") + self.isSuccess = block.isSuccess + self.resultMessage = block.resultMessage + return \ No newline at end of file diff --git a/model/ContractorInfo.py b/model/ContractorInfo.py new file mode 100644 index 0000000..edfa3cd --- /dev/null +++ b/model/ContractorInfo.py @@ -0,0 +1,65 @@ + + + +from mysql.connector import Error +import config +from datetime import datetime + + +class ContractorInfo: + def __init__(self, contractor_id): + self.ID = contractor_id + self.contInfo = None + self.fetchData() + + def fetchData(self): + """Fetch basic contractor info by ID.""" + try: + connection = config.get_db_connection() + with connection.cursor(dictionary=True, buffered=True) as cursor: + cursor.callproc('GetContractorInfoById', [self.ID]) + # Get the first result set + for result in cursor.stored_results(): + self.contInfo = result.fetchone() + except Error as e: + print(f"Error fetching contractor info: {e}") + finally: + if connection.is_connected(): + connection.close() + + def fetchalldata(self): + """Fetch hold types and invoices for contractor.""" + data = {} + try: + connection = config.get_db_connection() + with connection.cursor(dictionary=True, buffered=True) as cursor: + # Fetch Hold Types + cursor.callproc('GetHoldTypesByContractor', [self.ID]) + hold_types = [] + for result in cursor.stored_results(): + hold_types = result.fetchall() + hold_type_map = {ht['hold_type_id']: ht['hold_type'] for ht in hold_types} + data['hold_types'] = hold_type_map + + # Fetch Invoices + cursor.callproc('GetInvoicesByContractor', [self.ID]) + invoices = [] + for result in cursor.stored_results(): + invoices = result.fetchall() + + # Remove duplicate invoices + seen_ids = set() + unique_invoices = [] + for inv in invoices: + if inv['Invoice_Id'] not in seen_ids: + seen_ids.add(inv['Invoice_Id']) + unique_invoices.append(inv) + data['invoices'] = unique_invoices + + except Error as e: + print(f"Error fetching contractor data: {e}") + finally: + if connection.is_connected(): + connection.close() + + return data diff --git a/model/District.py b/model/District.py new file mode 100644 index 0000000..2901f3e --- /dev/null +++ b/model/District.py @@ -0,0 +1,80 @@ +from model.Utilities import ItemCRUDType +from model.ItemCRUD import ItemCRUD + +class District: + + isSuccess = False + resultMessage = "" + + def __init__(self): + self.isSuccess = False + self.resultMessage = "" + + # edit district + def EditDistrict(self, request, district_id): + district = ItemCRUD(itemType=ItemCRUDType.District) + + district_name = request.form['district_Name'].strip() + state_id = request.form['state_Id'] + + district.EditItem(request=request, childid=district_id, parentid=state_id, childname=district_name,storedprocupdate="UpdateDistrict" ) + self.isSuccess = district.isSuccess + self.resultMessage = district.resultMessage + return + + # add district + def AddDistrict(self, request): + + district = ItemCRUD(ItemCRUDType.District) + + district_name = request.form['district_Name'].strip() + state_id = request.form['state_Id'] + + district.AddItem(request=request, parentid=state_id, childname=district_name, storedprocfetch="GetDistrictByNameAndState", storedprocadd="SaveDistrict" ) + self.isSuccess = district.isSuccess + self.resultMessage = district.resultMessage + return + + + # get all district data + def GetAllDistricts(self, request): + district = ItemCRUD(itemType=ItemCRUDType.District) + districtsdata = district.GetAllData(request=request, storedproc="GetAllDistricts") + self.isSuccess = district.isSuccess + self.resultMessage = district.resultMessage + return districtsdata + + # check district validation + def CheckDistrict(self, request): + district = ItemCRUD(itemType=ItemCRUDType.District) + district_name = request.json.get('district_Name', '').strip() + state_id = request.json.get('state_Id', '') + result = district.CheckItem(request=request, parentid=state_id, childname=district_name, storedprocfetch="GetDistrictByNameAndState") + self.isSuccess = district.isSuccess + self.resultMessage = district.resultMessage + return result + + # get district by district id + def GetDistrictByID(self, request, district_id): + district = ItemCRUD(itemType=ItemCRUDType.District) + + districtdata = district.GetDataByID( + id=district_id, + storedproc="GetDistrictDataByID" + ) + + if districtdata: + self.isSuccess = True + else: + self.isSuccess = False + self.resultMessage = "District not found" + + return districtdata + + + # Delete District by district id + def DeleteDistrict(self, request, district_id): + district = ItemCRUD(itemType=ItemCRUDType.District) + district.DeleteItem(request=request,itemID=district_id,storedprocDelete="DeleteDistrict") + self.isSuccess = district.isSuccess + self.resultMessage = str(district.resultMessage) \ No newline at end of file diff --git a/model/FolderAndFile.py b/model/FolderAndFile.py new file mode 100644 index 0000000..30f5336 --- /dev/null +++ b/model/FolderAndFile.py @@ -0,0 +1,53 @@ +import os +from flask import current_app + + +class FolderAndFile: + + # ----------------------------- + # BASE FOLDER METHODS + # ----------------------------- + @staticmethod + def get_download_folder(): + folder = os.path.join(current_app.root_path, "static", "downloads") + + if not os.path.exists(folder): + os.makedirs(folder) + + os.makedirs(folder, exist_ok=True) + return folder + + @staticmethod + def get_upload_folder(): + folder = os.path.join(current_app.root_path, "static", "uploads") + + if not os.path.exists(folder): + os.makedirs(folder) + + os.makedirs(folder, exist_ok=True) + return folder + + @staticmethod + def get_logs_folder(): + folder = os.path.join(current_app.root_path, "logs") + + if not os.path.exists(folder): + os.makedirs(folder) + + os.makedirs(folder, exist_ok=True) + return folder + + # FILE PATH METHODS - download + @staticmethod + def get_download_path(filename): + return os.path.join(FolderAndFile.get_download_folder(), filename) + + # FILE PATH METHODS - upload file + @staticmethod + def get_upload_path(filename): + return os.path.join(FolderAndFile.get_upload_folder(), filename) + + # FILE PATH METHODS - activity log file + @staticmethod + def get_activity_log_path(filename): + return os.path.join(FolderAndFile.get_logs_folder(), filename) \ No newline at end of file diff --git a/model/GST.py b/model/GST.py new file mode 100644 index 0000000..71da41e --- /dev/null +++ b/model/GST.py @@ -0,0 +1,51 @@ +from model.ItemCRUD import ItemCRUD +from model.Utilities import ItemCRUDType + +class GST: + + @staticmethod + def get_unreleased_gst(): + # Use ItemCRUD for Invoices + invoice_crud = ItemCRUD(itemType=ItemCRUDType.Invoice) + invoices_rows = invoice_crud.GetAllData(storedproc="GetAllInvoicesBasic") + + if not invoice_crud.isSuccess: + return [] # Could also log invoice_crud.resultMessage + + invoices = [ + dict( + Invoice_No=row[1], + GST_SD_Amount=float(row[2]) if row[2] is not None else 0 + ) + for row in invoices_rows + ] + + # Use ItemCRUD for GST Releases + gst_crud = ItemCRUD(itemType=ItemCRUDType.GSTRelease) + gst_rows = gst_crud.GetAllData(storedproc="GetAllGSTReleasesBasic") + + if not gst_crud.isSuccess: + return [] # Could also log gst_crud.resultMessage + + gst_invoice_nos = { + g[2] # Invoice_No is at index 2 + for g in gst_rows + if g[2] + } + + gst_basic_amounts = { + float(g[3]) # Basic_Amount at index 3 + for g in gst_rows + if g[3] is not None + } + + # Filter unreleased invoices + unreleased = [] + for inv in invoices: + match_by_invoice = inv['Invoice_No'] in gst_invoice_nos + match_by_gst_amount = inv['GST_SD_Amount'] in gst_basic_amounts + + if not (match_by_invoice or match_by_gst_amount): + unreleased.append(inv) + + return unreleased \ No newline at end of file diff --git a/model/HoldTypes.py b/model/HoldTypes.py new file mode 100644 index 0000000..b8a9a46 --- /dev/null +++ b/model/HoldTypes.py @@ -0,0 +1,90 @@ +from flask import request +from model.ItemCRUD import ItemCRUD +from model.Utilities import ItemCRUDType + +class HoldTypes: + """CRUD operations for Hold Types using ItemCRUD""" + + def __init__(self): + self.isSuccess = False + self.resultMessage = "" + + # ------------------- Add Hold Type ------------------- + def AddHoldType(self, request): + hold = ItemCRUD(itemType=ItemCRUDType.HoldType) + hold_name = request.form.get('hold_type', '').strip() + + hold.AddItem( + request=request, + parentid=None, + childname=hold_name, + storedprocfetch="CheckHoldTypeExists", + storedprocadd="SaveHoldType" + ) + + self.isSuccess = hold.isSuccess + self.resultMessage = hold.resultMessage + + # ------------------- Get All Hold Types ------------------- + def GetAllHoldTypes(self, request=None): + hold = ItemCRUD(itemType=ItemCRUDType.HoldType) + rows = hold.GetAllData(request=request, storedproc="GetAllHoldTypes") + + self.isSuccess = hold.isSuccess + self.resultMessage = hold.resultMessage + + # ✅ Convert tuple → dictionary + data = [] + for row in rows: + data.append({ + "hold_type_id": row[0], + "hold_type": row[1] + }) + + return data + + # ------------------- Get Hold Type By ID ------------------- + def GetHoldTypeByID(self, hold_type_id): + hold = ItemCRUD(itemType=ItemCRUDType.HoldType) + row = hold.GetDataByID(hold_type_id, storedproc="GetHoldTypesById") + + self.isSuccess = hold.isSuccess + self.resultMessage = hold.resultMessage + + # ✅ Convert tuple → dictionary + if row: + return { + "hold_type_id": row[0], + "hold_type": row[1] + } + + return None + + # ------------------- Update Hold Type ------------------- + def EditHoldType(self, request, hold_type_id): + hold = ItemCRUD(itemType=ItemCRUDType.HoldType) + hold_name = request.form.get('hold_type', '').strip() + + hold.EditItem( + request=request, + childid=hold_type_id, + parentid=None, + childname=hold_name, + storedprocupdate="UpdateHoldTypeById" + ) + + self.isSuccess = hold.isSuccess + self.resultMessage = hold.resultMessage + + # ------------------- Delete Hold Type ------------------- + def DeleteHoldType(self, request, hold_type_id): + hold = ItemCRUD(itemType=ItemCRUDType.HoldType) + + hold.DeleteItem( + request=request, + itemID=hold_type_id, + storedprocDelete="DeleteHoldType" + ) + + self.isSuccess = hold.isSuccess + self.resultMessage = hold.resultMessage \ No newline at end of file diff --git a/model/Invoice.py b/model/Invoice.py new file mode 100644 index 0000000..0015e2f --- /dev/null +++ b/model/Invoice.py @@ -0,0 +1,237 @@ + +import config +import mysql.connector + +# ------------------- Helper Functions ------------------- +def clear_results(cursor): + """Consume all stored results to prevent cursor issues.""" + for r in cursor.stored_results(): + r.fetchall() + +def fetch_one(cursor): + """Fetch first row from stored results.""" + for result in cursor.stored_results(): + return result.fetchone() + return None + +def fetch_all(cursor): + """Fetch all rows from stored results.""" + data = [] + for result in cursor.stored_results(): + data.extend(result.fetchall()) + return data + +def get_numeric_values(data): + """Return numeric fields for invoices safely.""" + return [ + float(data.get('basic_amount') or 0), + float(data.get('debit_amount') or 0), + float(data.get('after_debit_amount') or 0), + float(data.get('amount') or 0), + float(data.get('gst_amount') or 0), + float(data.get('tds_amount') or 0), + float(data.get('sd_amount') or 0), + float(data.get('on_commission') or 0), + float(data.get('hydro_testing') or 0), + float(data.get('gst_sd_amount') or 0), + float(data.get('final_amount') or 0), + ] + +def execute_db_operation(operation_func): + """General DB operation wrapper with commit/rollback.""" + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True) + try: + result = operation_func(cursor) + connection.commit() + return result + except Exception: + connection.rollback() + raise + finally: + cursor.close() + connection.close() + + +# ------------------- Village Functions ------------------- +def get_village_id(village_name): + def operation(cursor): + cursor.callproc("GetVillageIdByName", (village_name,)) + return fetch_one(cursor) + return execute_db_operation(operation) + +def get_all_villages(): + def operation(cursor): + cursor.callproc("GetAllVillages") + return fetch_all(cursor) + return execute_db_operation(operation) + + +# ------------------- Invoice Functions ------------------- +def insert_invoice(data, village_id): + def operation(cursor): + # Insert invoice + cursor.callproc('InsertInvoice', [ + data.get('pmc_no'), + village_id, + data.get('work_type'), + data.get('invoice_details'), + data.get('invoice_date'), + data.get('invoice_no'), + *get_numeric_values(data), + data.get('subcontractor_id') + + ]) + invoice_row = fetch_one(cursor) + if not invoice_row: + raise Exception("Invoice ID not returned") + invoice_id = invoice_row.get('invoice_id') + + # # Insert inpayment + # cursor.callproc('InsertInpayment', [ + # data.get('pmc_no'), + # village_id, + # data.get('work_type'), + # data.get('invoice_details'), + # data.get('invoice_date'), + # data.get('invoice_no'), + # *get_numeric_values(data), + # data.get('subcontractor_id') + # ]) + # clear_results(cursor) + return invoice_id + + return execute_db_operation(operation) + +def get_all_invoice_details(): + def operation(cursor): + cursor.callproc('GetAllInvoiceDetails') + return fetch_all(cursor) + return execute_db_operation(operation) + +def get_invoice_by_id(invoice_id): + def operation(cursor): + cursor.callproc('GetInvoiceDetailsById', [invoice_id]) + invoice = fetch_one(cursor) + + cursor.callproc('GetHoldAmountsByInvoiceId', [invoice_id]) + hold_amounts = fetch_all(cursor) + + if invoice: + invoice["hold_amounts"] = hold_amounts + return invoice + return execute_db_operation(operation) + +def update_invoice(data, invoice_id): + def operation(cursor): + cursor.callproc("GetVillageIdByName", (data.get('village'),)) + village = fetch_one(cursor) + if not village: + raise Exception("Village not found") + village_id = village['Village_Id'] + + cursor.callproc('UpdateInvoice', [ + data.get('pmc_no'), + village_id, + data.get('work_type'), + data.get('invoice_details'), + data.get('invoice_date'), + data.get('invoice_no'), + *get_numeric_values(data), + invoice_id + ]) + clear_results(cursor) + execute_db_operation(operation) + +# def update_inpayment(data): +# def operation(cursor): +# cursor.callproc('UpdateInpayment', [ +# data.get('work_type'), +# data.get('invoice_details'), +# data.get('invoice_date'), +# *get_numeric_values(data), +# data.get('pmc_no'), +# data.get('invoice_no') +# ]) +# clear_results(cursor) +# execute_db_operation(operation) + +def delete_invoice_data(invoice_id, user_id): + def operation(cursor): + # Fetch PMC and Invoice_No from DB + cursor.callproc('GetInvoicePMCById', (invoice_id,)) + record = {} + for result in cursor.stored_results(): + record = result.fetchone() or {} + if not record: + raise Exception("Invoice not found") + + # Use exact DB keys + pmc_no = record['PMC_No'] + invoice_no = record['Invoice_No'] + + # Delete invoice + cursor.callproc("DeleteInvoice", (invoice_id,)) + clear_results(cursor) + + # Delete inpayment + # cursor.callproc('DeleteInpaymentByPMCInvoice', (pmc_no, invoice_no)) + # clear_results(cursor) + + execute_db_operation(operation) + + +# ------------------- Subcontractor Functions ------------------- +def assign_subcontractor(data, village_id): + def operation(cursor): + cursor.callproc('AssignSubcontractor', [ + data.get('pmc_no'), + data.get('subcontractor_id'), + village_id + ]) + clear_results(cursor) + execute_db_operation(operation) + + +# ------------------- Hold Types Functions ------------------- +def insert_hold_types(data, invoice_id): + def operation(cursor): + hold_types = data.getlist('hold_type[]') + hold_amounts = data.getlist('hold_amount[]') + + for hold_type, hold_amount in zip(hold_types, hold_amounts): + if not hold_type: + continue + + cursor.callproc('GetHoldTypeIdByName', [hold_type]) + hold_type_result = fetch_one(cursor) + + if not hold_type_result: + cursor.callproc('InsertHoldType', [hold_type, 0]) + cursor.execute("SELECT @_InsertHoldType_1") + hold_type_id = cursor.fetchone()[0] + else: + hold_type_id = hold_type_result['hold_type_id'] + + cursor.callproc('InsertInvoiceSubcontractorHold', [ + data.get('subcontractor_id'), + invoice_id, + hold_type_id, + float(hold_amount or 0) + ]) + clear_results(cursor) + execute_db_operation(operation) + +def get_all_hold_types(): + def operation(cursor): + cursor.callproc("GetAllHoldTypes") + return fetch_all(cursor) + return execute_db_operation(operation) + + +# ------------------- Contractor Functions ------------------- +def search_contractors(sub_query): + def operation(cursor): + cursor.callproc('SearchContractorsByName', [sub_query]) + return fetch_all(cursor) + return execute_db_operation(operation) \ No newline at end of file diff --git a/model/ItemCRUD.py b/model/ItemCRUD.py new file mode 100644 index 0000000..5ba53e3 --- /dev/null +++ b/model/ItemCRUD.py @@ -0,0 +1,397 @@ +from flask_login import current_user +from model.Utilities import RegEx, ResponseHandler, HtmlHelper, ItemCRUDType +from model.Log import LogHelper + +import config +import re +import mysql.connector + +# ---------------------------------------------------------- +# Mapping Class +# ---------------------------------------------------------- +class itemCRUDMapping: + + def __init__(self, itemType): + if itemType is ItemCRUDType.Village: + self.name = "Village" + elif itemType is ItemCRUDType.Block: + self.name = "Block" + elif itemType is ItemCRUDType.State: + self.name = "State" + elif itemType is ItemCRUDType.HoldType: + self.name = "Hold Type" + elif itemType is ItemCRUDType.Subcontractor: + self.name = "Subcontractor" + elif itemType.name == "GSTRelease": + self.name = "GSTRelease" + else: + self.name = "Item" + +# ---------------------------------------------------------- +# Generic CRUD Class +# ---------------------------------------------------------- +class ItemCRUD: + + def __init__(self, itemType): + self.isSuccess = False + self.resultMessage = "" + self.itemCRUDType = itemType + self.itemCRUDMapping = itemCRUDMapping(itemType) + + # ---------------------------------------------------------- + # DELETE + # ---------------------------------------------------------- + def DeleteItem(self, request, itemID, storedprocDelete): + + connection = config.get_db_connection() + cursor = connection.cursor() + + LogHelper.log_action( + f"Delete {self.itemCRUDMapping.name}", + f"User {current_user.id} deleted {self.itemCRUDMapping.name} '{itemID}'" + ) + + try: + cursor.callproc(storedprocDelete, (itemID,)) + connection.commit() + + self.isSuccess = True + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.delete_success(self.itemCRUDMapping.name), 200 + ) + + except mysql.connector.Error as e: + print(f"Error deleting {self.itemCRUDMapping.name}: {e}") + self.isSuccess = False + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.delete_failure(self.itemCRUDMapping.name), 500 + ) + + finally: + cursor.close() + connection.close() + + # ---------------------------------------------------------- + # ADD + # ---------------------------------------------------------- + def AddItem(self, request, parentid=None, childname=None, storedprocfetch=None, storedprocadd=None, data=None): + + connection = config.get_db_connection() + if not connection: + self.isSuccess = False + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.db_connection_failure(), 500 + ) + return + + cursor = connection.cursor() + + LogHelper.log_action( + f"Add {self.itemCRUDMapping.name}", + f"User {current_user.id} adding '{childname if childname else (data.get('Contractor_Name') if data else '')}'" + ) + + try: + # ====================================================== + # GSTRelease MULTI-FIELD + # ====================================================== + if self.itemCRUDType.name == "GSTRelease" and data: + + # Duplicate check (PMC_No + Invoice_No) + if storedprocfetch: + cursor.callproc(storedprocfetch, (data['PMC_No'], data['Invoice_No'])) + existing_item = None + for rs in cursor.stored_results(): + existing_item = rs.fetchone() + if existing_item: + self.isSuccess = False + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.already_exists(self.itemCRUDMapping.name), 409 + ) + return + + # Insert GSTRelease + cursor.callproc(storedprocadd, ( + data['PMC_No'], + data['Invoice_No'], + data['Basic_Amount'], + data['Final_Amount'], + data['Total_Amount'], + data['UTR'], + data['Contractor_ID'] + )) + connection.commit() + + self.isSuccess = True + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.add_success(self.itemCRUDMapping.name), 200 + ) + return + + # ====================================================== + # SUBCONTRACTOR MULTI-FIELD + # ====================================================== + if self.itemCRUDType.name == "Subcontractor" and data: + + cursor.callproc(storedprocfetch, (data['Contractor_Name'],)) + existing_item = None + for rs in cursor.stored_results(): + existing_item = rs.fetchone() + + if existing_item: + self.isSuccess = False + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.already_exists(self.itemCRUDMapping.name), 409 + ) + return + + cursor.callproc(storedprocadd, ( + data['Contractor_Name'], + data['Address'], + data['Mobile_No'], + data['PAN_No'], + data['Email'], + data['Gender'], + data['GST_Registration_Type'], + data['GST_No'], + data['Contractor_password'] + )) + connection.commit() + self.isSuccess = True + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.add_success(self.itemCRUDMapping.name), 200 + ) + return + + + # ====================================================== + # NORMAL SINGLE-FIELD (Village / Block / State) + # ====================================================== + if not re.match(RegEx.allPattern, childname): + self.isSuccess = False + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.invalid_name(self.itemCRUDMapping.name), 400 + ) + return + + if parentid is None: + cursor.callproc(storedprocfetch, (childname,)) + else: + cursor.callproc(storedprocfetch, (childname, parentid)) + + existing_item = None + for rs in cursor.stored_results(): + existing_item = rs.fetchone() + + if existing_item: + self.isSuccess = False + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.already_exists(self.itemCRUDMapping.name), 409 + ) + return + + if parentid is None: + cursor.callproc(storedprocadd, (childname,)) + else: + cursor.callproc(storedprocadd, (childname, parentid)) + + connection.commit() + self.isSuccess = True + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.add_success(self.itemCRUDMapping.name), 200 + ) + + except mysql.connector.Error as e: + print(f"Database Error: {e}") + self.isSuccess = False + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.add_failure(self.itemCRUDMapping.name), 500 + ) + + finally: + cursor.close() + connection.close() + + # ---------------------------------------------------------- + # EDIT + # ---------------------------------------------------------- + def EditItem(self, request, childid, parentid=None, childname=None, storedprocupdate=None, data=None): + + connection = config.get_db_connection() + cursor = connection.cursor() + + LogHelper.log_action( + f"Edit {self.itemCRUDMapping.name}", + f"User {current_user.id} edited '{childid}'" + ) + + try: + # ====================================================== + # GSTRelease MULTI-FIELD + # ====================================================== + if self.itemCRUDType.name == "GSTRelease" and data: + + cursor.callproc(storedprocupdate, ( + data['p_pmc_no'], # PMC_No + data['p_invoice_no'], # Invoice_No + data['p_basic_amount'], # Basic_Amount + data['p_final_amount'], # Final_Amount + data['p_total_amount'], # Total_Amount + data['p_utr'], # UTR + data['p_gst_release_id']# GST_Release_Id + )) + connection.commit() + + self.isSuccess = True + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.update_success(self.itemCRUDMapping.name), 200 + ) + return + + # ====================================================== + # SUBCONTRACTOR MULTI-FIELD + # ====================================================== + if self.itemCRUDType.name == "Subcontractor" and data: + + cursor.callproc(storedprocupdate, ( + childid, + data['Contractor_Name'], + data['Address'], + data['Mobile_No'], + data['PAN_No'], + data['Email'], + data['Gender'], + data['GST_Registration_Type'], + data['GST_No'], + data['Contractor_password'] + )) + connection.commit() + self.isSuccess = True + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.update_success(self.itemCRUDMapping.name), 200 + ) + return + + # ====================================================== + # NORMAL SINGLE-FIELD + # ====================================================== + if not re.match(RegEx.allPattern, childname): + self.isSuccess = False + self.resultMessage = ResponseHandler.update_failure(self.itemCRUDMapping.name)['message'] + return + + if parentid is None: + cursor.callproc(storedprocupdate, (childid, childname)) + else: + cursor.callproc(storedprocupdate, (childid, parentid, childname)) + + connection.commit() + self.isSuccess = True + self.resultMessage = ResponseHandler.update_success(self.itemCRUDMapping.name)['message'] + + except mysql.connector.Error as e: + print(f"Error updating {self.itemCRUDMapping.name}: {e}") + self.isSuccess = False + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.update_failure(self.itemCRUDMapping.name), 500 + ) + + finally: + cursor.close() + connection.close() + + # ---------------------------------------------------------- + # GET ALL + # ---------------------------------------------------------- + def GetAllData(self, request, storedproc): + + data = [] + connection = config.get_db_connection() + if not connection: + return [] + + cursor = connection.cursor() + try: + cursor.callproc(storedproc) + for result in cursor.stored_results(): + data = result.fetchall() + self.isSuccess = True + except mysql.connector.Error as e: + print(f"Error fetching {self.itemCRUDMapping.name}: {e}") + self.isSuccess = False + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.fetch_failure(self.itemCRUDMapping.name), 500 + ) + return [] + finally: + cursor.close() + connection.close() + + return data + + # ---------------------------------------------------------- + # GET BY ID + # ---------------------------------------------------------- + def GetDataByID(self, id, storedproc): + + data = None + connection = config.get_db_connection() + cursor = connection.cursor() + + try: + cursor.callproc(storedproc, (id,)) + for rs in cursor.stored_results(): + data = rs.fetchone() + except mysql.connector.Error as e: + print(f"Error fetching {self.itemCRUDMapping.name}: {e}") + finally: + cursor.close() + connection.close() + + return data + + # ---------------------------------------------------------- + # CHECK ITEM + # ---------------------------------------------------------- + def CheckItem(self, request, parentid, childname, storedprocfetch): + + connection = config.get_db_connection() + cursor = connection.cursor() + + LogHelper.log_action( + f"Check {self.itemCRUDMapping.name}", + f"User {current_user.id} checked '{childname}'" + ) + + if not re.match(RegEx.allPattern, childname): + return HtmlHelper.json_response( + ResponseHandler.invalid_name(self.itemCRUDMapping.name), 400 + ) + + try: + if parentid is None: + cursor.callproc(storedprocfetch, (childname,)) + else: + cursor.callproc(storedprocfetch, (childname, parentid)) + + existing_item = None + for rs in cursor.stored_results(): + existing_item = rs.fetchone() + + if existing_item: + return HtmlHelper.json_response( + ResponseHandler.already_exists(self.itemCRUDMapping.name), 409 + ) + + return HtmlHelper.json_response( + ResponseHandler.is_available(self.itemCRUDMapping.name), 200 + ) + + except mysql.connector.Error as e: + print(f"Error checking {self.itemCRUDMapping.name}: {e}") + return HtmlHelper.json_response( + ResponseHandler.fetch_failure(self.itemCRUDMapping.name), 500 + ) + finally: + cursor.close() + connection.close() \ No newline at end of file diff --git a/model/Log.py b/model/Log.py new file mode 100644 index 0000000..d3ed22f --- /dev/null +++ b/model/Log.py @@ -0,0 +1,84 @@ +import os +from flask import current_app +from flask_login import LoginManager, UserMixin, login_user, logout_user, login_required, current_user + +from datetime import datetime + +from model.FolderAndFile import FolderAndFile + +class LogHelper: + @staticmethod + def log_action(action, details=""): + """Log user actions with timestamp, user, action, and details.""" + logData = LogData() + logData.WriteLog(action, details="") + +class LogData: + filepath = "" + timestamp = None + + def __init__(self): + self.filepath = FolderAndFile.get_activity_log_path('activity.log') + self.timestamp = datetime.now().strftime("%Y-%m-%d %H:%M:%S") + self.user = LogData.get_current_user() + + + @staticmethod + def get_current_user(): + if hasattr(current_user, "cn") and current_user.cn: + return current_user.cn + elif hasattr(current_user, "username") and current_user.username: + return current_user.username + elif hasattr(current_user, "sAMAccountName") and current_user.sAMAccountName: + return current_user.sAMAccountName + return "Unknown" + + def WriteLog(self, action, details=""): + """Log user actions with timestamp, user, action, and details.""" + + with open(self.filepath, "a", encoding="utf-8") as f: + f.write( + f"Timestamp: {self.timestamp} | " + f"User: {self.user} | " + f"Action: {action} | " + f"Details: {details}\n" + ) + + def GetActivitiesLog(self): + logs = [] + + if os.path.exists(self.filepath): + with open(self.filepath, 'r') as f: + for line in f: + parts = line.strip().split(" | ") + if len(parts) == 4: + logs.append({ + "timestamp": parts[0].replace("Timestamp:", "").strip(), + "user": parts[1].replace("User:", "").strip(), + "action": parts[2].replace("Action:", "").strip(), + "details": parts[3].replace("Details:", "").strip() + }) + return logs + + def GetFilteredActivitiesLog(self, startDate, endDate, userName): + filtered_logs = self.GetActivitiesLog() + + # Date filter + if startDate or endDate: + try: + start_dt = datetime.strptime(startDate, "%Y-%m-%d") if startDate else datetime.min + end_dt = datetime.strptime(endDate, "%Y-%m-%d") if endDate else datetime.max + + filtered_logs = [ + log for log in filtered_logs + if start_dt <= datetime.strptime(log["timestamp"], "%Y-%m-%d %H:%M:%S") <= end_dt + ] + + except Exception as e: + print("Date filter error:", e) + + # Username filter + if userName: + filtered_logs = [log for log in filtered_logs if userName.lower() in log["user"].lower()] + + return filtered_logs \ No newline at end of file diff --git a/model/PmcReport.py b/model/PmcReport.py new file mode 100644 index 0000000..c8ec69e --- /dev/null +++ b/model/PmcReport.py @@ -0,0 +1,456 @@ +import openpyxl +from openpyxl.styles import Font, PatternFill +import config +from flask_login import current_user +from model.Log import LogHelper + +from model.Report import ReportHelper +from model.FolderAndFile import FolderAndFile + +class PmcReport: + + @staticmethod + def get_pmc_report(pmc_no): + + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True, buffered=True) + + try: + + # cursor.callproc("GetContractorInfoByPmcNo", (pmc_no,)) + # pmc_info = next(cursor.stored_results()).fetchone() + pmc_info = ReportHelper.execute_sp(cursor, 'GetContractorInfoByPmcNo', [pmc_no], True) + + if not pmc_info: + return None + + cursor.callproc("Get_pmc_hold_types", (pmc_no, pmc_info["Contractor_Id"])) + hold_types = next(cursor.stored_results()).fetchall() + + # Extract hold_type_ids + hold_type_ids = [ht['hold_type_id'] for ht in hold_types] + + invoices = [] + hold_amount_total = 0 + if hold_type_ids: + hold_type_ids_str = ",".join(map(str, hold_type_ids)) + cursor.callproc( + 'GetInvoices_WithHold', + [pmc_no, pmc_info["Contractor_Id"], hold_type_ids_str] + ) + else: + cursor.callproc( + 'GetInvoices_NoHold', + [pmc_no, pmc_info["Contractor_Id"]] + ) + for result in cursor.stored_results(): + invoices = result.fetchall() + + if hold_type_ids: + hold_amount_total = sum(row.get('hold_amount', 0) or 0 for row in invoices) + + total_invo_final = sum(row.get('Final_Amount', 0) or 0 for row in invoices) + + + # GST RELEASE + # cursor.callproc('GetGSTReleaseByPMC', [pmc_no]) + # gst_rel = [] + # for result in cursor.stored_results(): + # gst_rel = result.fetchall() + + gst_rel = ReportHelper.execute_sp(cursor, 'GetGSTReleaseByPMC', [pmc_no]) + + total_gst_basic = sum(row.get('basic_amount', 0) or 0 for row in gst_rel) + total_gst_final = sum(row.get('final_amount', 0) or 0 for row in gst_rel) + + # ---------------- HOLD RELEASE ---------------- + # cursor.callproc('GetHoldReleaseByPMC', [pmc_no]) + # hold_release = [] + # for result in cursor.stored_results(): + # hold_release = result.fetchall() + + hold_release = ReportHelper.execute_sp(cursor, 'GetHoldReleaseByPMC', [pmc_no]) + + # ---------------- CREDIT NOTE ---------------- + # cursor.callproc('GetCreditNoteByPMC', [pmc_no]) + # credit_note = [] + # for result in cursor.stored_results(): + # credit_note = result.fetchall() + + credit_note = ReportHelper.execute_sp(cursor, 'GetCreditNoteByPMC', [pmc_no]) + + payments = ReportHelper.execute_sp(cursor, 'GetPaymentsByPMC', [pmc_no]) + + + # ---------------- PAYMENTS ---------------- + # cursor.callproc('GetPaymentsByPMC', [pmc_no]) + # payments = [] + # for result in cursor.stored_results(): + # payments = result.fetchall() + + + + total_pay_amount = sum(row.get('Payment_Amount', 0) or 0 for row in payments) + total_pay_total = sum(row.get('Total_amount', 0) or 0 for row in payments) + + totals = { + "sum_invo_basic_amt": sum(row.get('Basic_Amount', 0) or 0 for row in invoices), + "sum_invo_debit_amt": sum(row.get('Debit_Amount', 0) or 0 for row in invoices), + "sum_invo_after_debit_amt": sum(row.get('After_Debit_Amount', 0) or 0 for row in invoices), + "sum_invo_amt": sum(row.get('Amount', 0) or 0 for row in invoices), + "sum_invo_gst_amt": sum(row.get('GST_Amount', 0) or 0 for row in invoices), + "sum_invo_tds_amt": sum(row.get('TDS_Amount', 0) or 0 for row in invoices), + "sum_invo_ds_amt": sum(row.get('SD_Amount', 0) or 0 for row in invoices), + "sum_invo_on_commission": sum(row.get('On_Commission', 0) or 0 for row in invoices), + "sum_invo_hydro_test": sum(row.get('Hydro_Testing', 0) or 0 for row in invoices), + "sum_invo_gst_sd_amt": sum(row.get('GST_SD_Amount', 0) or 0 for row in invoices), + "sum_invo_final_amt": total_invo_final, + "sum_invo_hold_amt": hold_amount_total, + "sum_gst_basic_amt": total_gst_basic, + "sum_gst_final_amt": total_gst_final, + "sum_pay_payment_amt": total_pay_amount, + "sum_pay_tds_payment_amt": sum(row.get('TDS_Payment_Amount', 0) or 0 for row in payments), + "sum_pay_total_amt": total_pay_total + } + + return { + "info": pmc_info, + "invoices": invoices, + "hold_types": hold_types, + "gst_rel": gst_rel, + "payments": payments, + "credit_note": credit_note, + "hold_release": hold_release, + "total": totals + } + + finally: + cursor.close() + connection.close() + + + @staticmethod + def download_pmc_report(pmc_no): + + connection = config.get_db_connection() + if not connection: + return None + + cursor = connection.cursor(dictionary=True) + + try: + # filename + filename = f"PMC_Report_{pmc_no}.xlsx" + + output_folder = FolderAndFile.get_download_folder() + output_file = FolderAndFile.get_download_path(filename) + + # ================= DATA FETCH ================= + + contractor_info = ReportHelper.execute_sp(cursor, 'GetContractorDetailsByPMC', [pmc_no], "one") + + if not contractor_info: + return None + + hold_types = ReportHelper.execute_sp(cursor, 'GetHoldTypesByContractor', [contractor_info["Contractor_Id"]]) + + hold_type_map = {ht['hold_type_id']: ht['hold_type'] for ht in hold_types} + + invoices = ReportHelper.execute_sp(cursor, 'GetInvoicesAndGstReleaseByPmcNo', [pmc_no]) + + credit_notes = ReportHelper.execute_sp(cursor, 'GetCreditNoteByContractor', [contractor_info["Contractor_Id"]]) + + hold_amounts = ReportHelper.execute_sp(cursor, 'GetHoldAmountsByContractor', [contractor_info["Contractor_Id"]]) + + all_payments = ReportHelper.execute_sp(cursor, 'GetAllPaymentsByPMC', [pmc_no]) + + gst_releases = ReportHelper.execute_sp(cursor, 'GetGSTReleaseDetailsByPMC', [pmc_no]) + + # ================= DATA MAPPING ================= + + hold_data = {} + for h in hold_amounts: + hold_data.setdefault(h['Invoice_Id'], {})[h['hold_type_id']] = h['hold_amount'] + + payments_map = {} + for pay in all_payments: + if pay['invoice_no']: + payments_map.setdefault(pay['invoice_no'], []).append(pay) + + # ================= LOG ================= + LogHelper.log_action( + "Download PMC Report", + f"User {current_user.id} Download PMC Report '{pmc_no}'" + ) + + # ================= EXCEL ================= + workbook = openpyxl.Workbook() + sheet = workbook.active + sheet.title = "PMC Report" + + # HEADER INFO + sheet.append(["", "", "Laxmi Civil Engineering Services PVT. LTD."]) + sheet.append(["Contractor Name", contractor_info["Contractor_Name"]]) + sheet.append(["State", contractor_info["State_Name"]]) + sheet.append(["District", contractor_info["District_Name"]]) + sheet.append(["Block", contractor_info["Block_Name"]]) + sheet.append([]) + + base_headers = [ + "PMC No","Village","Work Type","Invoice Details","Invoice Date","Invoice No", + "Basic Amount","Debit","After Debit Amount","GST","Amount","TDS", + "SD","On Commission","Hydro Testing","GST SD Amount" + ] + + hold_headers = [ht['hold_type'] for ht in hold_types] + + payment_headers = [ + "Final Amount","Payment Amount","TDS Payment","Total Paid","UTR" + ] + + headers = base_headers + hold_headers + payment_headers + sheet.append(headers) + + # STYLE + for cell in sheet[sheet.max_row]: + cell.font = Font(bold=True) + + # DATA + seen_invoices = set() + + for inv in invoices: + + invoice_no = inv["Invoice_No"] + payments = payments_map.get(invoice_no, []) + + if invoice_no in seen_invoices: + continue + + seen_invoices.add(invoice_no) + + first_payment = payments[0] if payments else None + + row = [ + pmc_no, + inv["Village_Name"], + inv["Work_Type"], + inv["Invoice_Details"], + inv["Invoice_Date"], + invoice_no, + inv["Basic_Amount"], + inv["Debit_Amount"], + inv["After_Debit_Amount"], + inv["GST_Amount"], + inv["Amount"], + inv["TDS_Amount"], + inv["SD_Amount"], + inv["On_Commission"], + inv["Hydro_Testing"], + inv["GST_SD_Amount"] + ] + + # HOLD DATA + invoice_holds = hold_data.get(inv["Invoice_Id"], {}) + for ht_id in hold_type_map.keys(): + row.append(invoice_holds.get(ht_id, "")) + + # PAYMENT DATA + row += [ + inv["Final_Amount"], + first_payment["Payment_Amount"] if first_payment else "", + first_payment["TDS_Payment_Amount"] if first_payment else "", + first_payment["Total_amount"] if first_payment else "", + first_payment["UTR"] if first_payment else "" + ] + + sheet.append(row) + + # AUTO WIDTH + for col in sheet.columns: + max_len = max((len(str(cell.value)) for cell in col if cell.value), default=0) + sheet.column_dimensions[col[0].column_letter].width = max_len + 2 + + # SAVE + workbook.save(output_file) + workbook.close() + + return output_folder, filename + + except Exception as e: + print(f"Error generating PMC report: {e}") + return None + + finally: + cursor.close() + connection.close() + + # @staticmethod + # def download_pmc_report(pmc_no): + + # connection = config.get_db_connection() + # cursor = connection.cursor(dictionary=True) + + # # output_folder = "static/download" + # # output_file = os.path.join(output_folder, f"PMC_Report_{pmc_no}.xlsx") + # output_folder = FolderAndFile.get_download_folder + # filename = f"PMC_Report_{pmc_no}.xlsx" + # output_file = FolderAndFile.get_download_path(filename) + + # try: + + # cursor.callproc('GetContractorDetailsByPMC', [pmc_no]) + # contractor_info = next(cursor.stored_results()).fetchone() + + # if not contractor_info: + # return None + + # cursor.callproc('GetHoldTypesByContractor', [contractor_info["Contractor_Id"]]) + # hold_types = next(cursor.stored_results()).fetchall() + + # hold_type_map = {ht['hold_type_id']: ht['hold_type'] for ht in hold_types} + + # cursor.callproc('GetInvoicesAndGstReleaseByPmcNo', [pmc_no]) + # invoices = next(cursor.stored_results()).fetchall() + + # cursor.callproc('GetCreditNoteByContractor',[contractor_info["Contractor_Id"]]) + + # credit_notes = [] + # for result in cursor.stored_results(): + # credit_notes = result.fetchall() + + # credit_note_map = {} + # for cn in credit_notes: + # key = (cn["PMC_No"], cn["Invoice_No"]) + # credit_note_map.setdefault(key, []).append(cn) + + # cursor.callproc('GetHoldAmountsByContractor', [contractor_info["Contractor_Id"]]) + # hold_amounts = next(cursor.stored_results()).fetchall() + + # hold_data = {} + # for h in hold_amounts: + # hold_data.setdefault(h['Invoice_Id'], {})[h['hold_type_id']] = h['hold_amount'] + + # cursor.callproc('GetAllPaymentsByPMC', [pmc_no]) + # all_payments = next(cursor.stored_results()).fetchall() + + # payments_map = {} + # extra_payments = [] + + # for pay in all_payments: + # if pay['invoice_no']: + # payments_map.setdefault(pay['invoice_no'], []).append(pay) + # else: + # extra_payments.append(pay) + + # # ---------------- GST RELEASE DETAILS ---------------- + # cursor.callproc('GetGSTReleaseDetailsByPMC', [pmc_no]) + + # gst_releases = [] + # for result in cursor.stored_results(): + # gst_releases = result.fetchall() + + # gst_release_map = {} + + # for gr in gst_releases: + + # invoice_nos = [] + + # if gr['Invoice_No']: + + # cleaned = gr['Invoice_No'].replace(' ', '') + + # if '&' in cleaned: + # invoice_nos = cleaned.split('&') + + # elif ',' in cleaned: + # invoice_nos = cleaned.split(',') + + # else: + # invoice_nos = [cleaned] + + # for inv_no in invoice_nos: + # gst_release_map.setdefault(inv_no, []).append(gr) + + # LogHelper.log_action( + # "Download PMC Report", + # f"User {current_user.id} Download PMC Report '{pmc_no}'" + # ) + + # workbook = openpyxl.Workbook() + # sheet = workbook.active + # sheet.title = "PMC Report" + + # sheet.append(["", "", "Laxmi Civil Engineering Services PVT. LTD."]) + # sheet.append(["Contractor Name", contractor_info["Contractor_Name"], "", "GST No", contractor_info["GST_No"], "", "GST Type", contractor_info["GST_Registration_Type"]]) + # sheet.append(["State", contractor_info["State_Name"], "", "PAN No", contractor_info["PAN_No"], "", "Address", contractor_info["Address"]]) + # sheet.append(["District", contractor_info["District_Name"], "", "Mobile No", contractor_info["Mobile_No"]]) + # sheet.append(["Block", contractor_info["Block_Name"], "", "Email", contractor_info["Email"]]) + # sheet.append([]) + + # base_headers = [ + # "PMC No","Village","Work Type","Invoice Details","Invoice Date","Invoice No", + # "Basic Amount","Debit","After Debit Amount","GST (18%)","Amount","TDS (1%)", + # "SD (5%)","On Commission","Hydro Testing","GST SD Amount" + # ] + + # hold_headers = [ht['hold_type'] for ht in hold_types] + + # payment_headers = [ + # "Final Amount","Payment Amount","TDS Payment","Total Paid","UTR" + # ] + + # sheet.append(base_headers + hold_headers + payment_headers) + + # header_fill = PatternFill(start_color="ADD8E6",end_color="ADD8E6",fill_type="solid") + # header_font = Font(bold=True) + + # for cell in sheet[sheet.max_row]: + # cell.font = header_font + # cell.fill = header_fill + + # seen_invoices = set() + # processed_payments = set() + + # for inv in invoices: + + # invoice_no = inv["Invoice_No"] + # payments = payments_map.get(invoice_no, []) + + # if invoice_no not in seen_invoices: + + # seen_invoices.add(invoice_no) + # first_payment = payments[0] if payments else None + + # row = [ + # pmc_no, inv["Village_Name"], inv["Work_Type"], + # inv["Invoice_Details"], inv["Invoice_Date"], invoice_no, + # inv["Basic_Amount"], inv["Debit_Amount"], + # inv["After_Debit_Amount"], inv["GST_Amount"], + # inv["Amount"], inv["TDS_Amount"], inv["SD_Amount"], + # inv["On_Commission"], inv["Hydro_Testing"], inv["GST_SD_Amount"] + # ] + + # invoice_holds = hold_data.get(inv["Invoice_Id"], {}) + + # for ht_id in hold_type_map.keys(): + # row.append(invoice_holds.get(ht_id, "")) + + # row += [ + # inv["Final_Amount"], + # first_payment["Payment_Amount"] if first_payment else "", + # first_payment["TDS_Payment_Amount"] if first_payment else "", + # first_payment["Total_amount"] if first_payment else "", + # first_payment["UTR"] if first_payment else "" + # ] + + # sheet.append(row) + + # workbook.save(output_file) + # workbook.close() + + # return output_folder, filename + + # finally: + + # cursor.close() + # connection.close() \ No newline at end of file diff --git a/model/Report.py b/model/Report.py new file mode 100644 index 0000000..6f16977 --- /dev/null +++ b/model/Report.py @@ -0,0 +1,382 @@ +import config +from datetime import datetime +from flask import send_file +import os +import openpyxl +from openpyxl.styles import Font, PatternFill + +from model.FolderAndFile import FolderAndFile + +class ReportHelper: + isSuccess = False + resultMessage = "" + data=[] + + def __init__(self): + self.isSuccess = False + self.resultMessage = "" + self.data = [] + + @staticmethod + def execute_sp(cursor, proc_name, params=[], fetch_one=False): + cursor.callproc(proc_name, params) + return ( + ReportHelper.fetch_one_result(cursor) + if fetch_one else + ReportHelper.fetch_all_results(cursor) + ) + + @staticmethod + def fetch_all_results(cursor): + data = [] + for result in cursor.stored_results(): + data = result.fetchall() + return data + + + @staticmethod + def fetch_one_result(cursor): + data = None + for result in cursor.stored_results(): + data = result.fetchone() + return data + + + @staticmethod + def search_contractor(request): + subcontractor_name = request.form.get("subcontractor_name") + pmc_no = request.form.get("pmc_no") + state = request.form.get("state") + district = request.form.get("district") + block = request.form.get("block") + village = request.form.get("village") + year_from = request.form.get("year_from") + year_to = request.form.get("year_to") + + connection = config.get_db_connection() + if not connection: + return [] + + cursor = connection.cursor(dictionary=True) + + try: + data = ReportHelper.execute_sp( + cursor, + "search_contractor_info", + [ + subcontractor_name or None, + pmc_no or None, + state or None, + district or None, + block or None, + village or None, + year_from or None, + year_to or None + ] + ) + + except Exception as e: + print(f"Error in search_contractor: {e}") + data = [] + + finally: + cursor.close() + connection.close() + + return data + + + @staticmethod + def get_contractor_report(contractor_id): + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True, buffered=True) + + try: + # Contractor Info (only one fetch) + contInfo = ReportHelper.execute_sp(cursor, 'GetContractorInfo', [contractor_id], True) + # Hold Types + hold_types = ReportHelper.execute_sp(cursor, 'GetContractorHoldTypes', [contractor_id]) + # Invoices + invoices = ReportHelper.execute_sp(cursor, 'GetContractorInvoices', [contractor_id]) + # GST Release + gst_rel = ReportHelper.execute_sp(cursor, 'GetGSTRelease', [contractor_id]) + # Hold Release + hold_release = ReportHelper.execute_sp(cursor, 'GetHoldRelease', [contractor_id]) + # Credit Note + credit_note = ReportHelper.execute_sp(cursor, 'GetCreditNote', [contractor_id]) + # Payments + payments = ReportHelper.execute_sp(cursor, 'GetPayments', [contractor_id]) + + # Totals + total = { + "sum_invo_basic_amt": float(sum(row['Basic_Amount'] or 0 for row in invoices)), + "sum_invo_debit_amt": float(sum(row['Debit_Amount'] or 0 for row in invoices)), + "sum_invo_after_debit_amt": float(sum(row['After_Debit_Amount'] or 0 for row in invoices)), + "sum_invo_amt": float(sum(row['Amount'] or 0 for row in invoices)), + "sum_invo_gst_amt": float(sum(row['GST_Amount'] or 0 for row in invoices)), + "sum_invo_tds_amt": float(sum(row['TDS_Amount'] or 0 for row in invoices)), + "sum_invo_ds_amt": float(sum(row['SD_Amount'] or 0 for row in invoices)), + "sum_invo_on_commission": float(sum(row['On_Commission'] or 0 for row in invoices)), + "sum_invo_hydro_test": float(sum(row['Hydro_Testing'] or 0 for row in invoices)), + "sum_invo_gst_sd_amt": float(sum(row['GST_SD_Amount'] or 0 for row in invoices)), + "sum_invo_final_amt": float(sum(row['Final_Amount'] or 0 for row in invoices)), + "sum_invo_hold_amt": float(sum(row['hold_amount'] or 0 for row in invoices)), + + "sum_gst_basic_amt": float(sum(row['basic_amount'] or 0 for row in gst_rel)), + "sum_gst_final_amt": float(sum(row['final_amount'] or 0 for row in gst_rel)), + + "sum_pay_payment_amt": float(sum(row['Payment_Amount'] or 0 for row in payments)), + "sum_pay_tds_payment_amt": float(sum(row['TDS_Payment_Amount'] or 0 for row in payments)), + "sum_pay_total_amt": float(sum(row['Total_amount'] or 0 for row in payments)) + } + + current_date = datetime.now().strftime('%Y-%m-%d') + + finally: + cursor.close() + connection.close() + + return { + "contInfo": contInfo, + "invoices": invoices, + "hold_types": hold_types, + "gst_rel": gst_rel, + "payments": payments, + "credit_note": credit_note, + "hold_release": hold_release, + "total": total, + "current_date": current_date + } + + + @staticmethod + def get_contractor_info(contractor_id): + from model.ContractorInfo import ContractorInfo + contractor = ContractorInfo(contractor_id) + return contractor.contInfo if contractor.contInfo else None + + # @staticmethod + # def generate_excel(contractor_id, contInfo, invoices, hold_types, hold_data, + # extra_payments_map, credit_note_map, gst_release_map, output_file): + @staticmethod + def generate_excel(contractor_id, contInfo, invoices, hold_types, hold_data, + credit_note_map, gst_release_map, output_file): + + workbook = openpyxl.Workbook() + sheet = workbook.active + sheet.title = "Contractor Report" + + # Contractor Info + for field, value in contInfo.items(): + sheet.append([field.replace("_", " "), value]) + sheet.append([]) + + # Headers + base_headers = ["PMC No", "Village", "Work Type", "Invoice Details", "Invoice Date", "Invoice No", + "Basic Amount", "Debit", "After Debit Amount", "GST (18%)", "Amount", "TDS (1%)", + "SD (5%)", "On Commission", "Hydro Testing", "GST SD Amount"] + + hold_headers = [ht['hold_type'] for ht in hold_types] + + payment_headers = ["Final Amount", "Payment Amount", "TDS Payment", "Total Paid", "UTR"] + + all_headers = base_headers + hold_headers + payment_headers + sheet.append(all_headers) + + for cell in sheet[sheet.max_row]: + cell.font = Font(bold=True) + cell.fill = PatternFill(start_color="ADD8E6", end_color="ADD8E6", fill_type="solid") + + processed_gst_releases = set() + appended_credit_keys = set() + previous_pmc_no = None + + for inv in invoices: + pmc_no = str(inv["PMC_No"]).strip() + + invoice_no = ( + inv["invoice_no"].replace(" ", "") if inv["invoice_no"] else "" + if inv["invoice_no"] not in (None, "", 0) + else "" + ) + + key = (pmc_no, invoice_no) + + # Yellow separator + if previous_pmc_no and pmc_no != previous_pmc_no: + sheet.append([""] * len(all_headers)) + yellow_fill = PatternFill(start_color="FFFF99", end_color="FFFF99", fill_type="solid") + for cell in sheet[sheet.max_row]: + cell.fill = yellow_fill + + previous_pmc_no = pmc_no + + # Invoice Row + row = [ + pmc_no, + inv.get("Village_Name", ""), + inv.get("Work_Type", ""), + inv.get("Invoice_Details", ""), + inv.get("Invoice_Date", ""), + invoice_no, + inv.get("Basic_Amount", ""), + inv.get("Debit_Amount", ""), + inv.get("After_Debit_Amount", ""), + inv.get("GST_Amount", ""), + inv.get("Amount", ""), + inv.get("TDS_Amount", ""), + inv.get("SD_Amount", ""), + inv.get("On_Commission", ""), + inv.get("Hydro_Testing", ""), + inv.get("GST_SD_Amount", "") + ] + + # Hold values + invoice_holds = hold_data.get(inv["Invoice_Id"], {}) + for ht_id in [ht['hold_type_id'] for ht in hold_types]: + row.append(invoice_holds.get(ht_id, "")) + + # Payment values + row += [ + inv.get("Final_Amount", ""), + inv.get("Payment_Amount", ""), + inv.get("TDS_Payment_Amount", ""), + inv.get("Total_Amount", ""), + inv.get("UTR", "") + ] + + sheet.append(row) + + # # Extra Payments + # if pmc_no in extra_payments_map: + # for ep in extra_payments_map[pmc_no]: + # extra_row = [pmc_no] + [""] * (len(base_headers) - 1) + # extra_row += [""] * len(hold_headers) + # extra_row += [ + # "", + # ep.get("Payment_Amount", ""), + # ep.get("TDS_Payment_Amount", ""), + # ep.get("Total_Amount", ""), + # ep.get("utr", "") + # ] + # sheet.append(extra_row) + + # del extra_payments_map[pmc_no] + + # GST Releases + if key in gst_release_map and key not in processed_gst_releases: + for gr in gst_release_map[key]: + gst_row = [ + pmc_no, "", "", "GST Release Note", "", gr.get("Invoice_No", ""), + gr.get("Basic_Amount", ""), "", "", "", "", "", "", "", "", "" + ] + + gst_row += [""] * len(hold_headers) + + gst_row += [ + gr.get("Final_Amount", ""), + "", + "", + gr.get("Total_Amount", ""), + gr.get("UTR", "") + ] + + sheet.append(gst_row) + + processed_gst_releases.add(key) + + # Credit Notes + if key in credit_note_map and key not in appended_credit_keys: + for cn in credit_note_map[key]: + cn_row = [ + pmc_no, "", "", cn.get("Invoice_Details", "Credit Note"), "", + cn.get("Invoice_No", ""), + cn.get("Basic_Amount", ""), + cn.get("Debit_Amount", ""), + cn.get("After_Debit_Amount", ""), + cn.get("GST_Amount", ""), + cn.get("Amount", ""), + "", "", "", "", "" + ] + + cn_row += [""] * len(hold_headers) + + cn_row += [ + cn.get("Final_Amount", ""), + "", + "", + cn.get("Total_Amount", ""), + cn.get("UTR", "") + ] + + sheet.append(cn_row) + + appended_credit_keys.add(key) + + # SAVE ONCE AT END + workbook.save(output_file) + + + @staticmethod + def create_contractor_report(contractor_id): + DOWNLOAD_FOLDER = os.path.join("static", "download") + os.makedirs(DOWNLOAD_FOLDER, exist_ok=True) + output_file = os.path.join(DOWNLOAD_FOLDER, f"Contractor_Report_{contractor_id}.xlsx") + + # Fetch Data + contInfo = ReportHelper.get_contractor_info(contractor_id) + if not contInfo: + return None, "No contractor found" + + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True, buffered=True) + + hold_types = ReportHelper.execute_sp(cursor, 'HoldTypesByContractorId', [contractor_id]) + invoices = ReportHelper.execute_sp(cursor, 'FetchInvoicesByContractor', [contractor_id]) + hold_amounts = ReportHelper.execute_sp(cursor, 'HoldAmountsByContractorId', [contractor_id]) + hold_data = {} + for h in hold_amounts: + hold_data.setdefault(h['Invoice_Id'], {})[h['hold_type_id']] = h['hold_amount'] + # # # -------- Extra Payments MAP -------- + # # extra_payments_raw = ReportHelper.execute_sp(cursor, 'GetExtraPayments') + # # extra_payments_map = {} + # # for ep in extra_payments_raw: + # # pmc = str(ep['pmc_no']).strip() + # # extra_payments_map.setdefault(pmc, []).append(ep) + + # -------- Credit Note MAP -------- + credit_note_raw = ReportHelper.execute_sp(cursor, 'GetCreditNotesByContractor', [contractor_id]) + credit_note_map = {} + for cn in credit_note_raw: + key = ( + str(cn['PMC_No']).strip(), + str(cn['Invoice_No']).replace(" ", "") if cn['Invoice_No'] else "" + ) + credit_note_map.setdefault(key, []).append(cn) + + # -------- GST MAP -------- + gst_release_raw = ReportHelper.execute_sp(cursor, 'GstReleasesByContractorId', [contractor_id]) + gst_release_map = {} + for gr in gst_release_raw: + key = ( + str(gr['PMC_No']).strip(), + str(gr['Invoice_No']).replace(" ", "") if gr['Invoice_No'] else "" + ) + gst_release_map.setdefault(key, []).append(gr) + + print("GST MAP:", gst_release_map) + # Generate Excel + # ReportHelper.generate_excel( + # contractor_id, contInfo, invoices, hold_types, hold_data, + # extra_payments_map, credit_note_map, gst_release_map, output_file + # ) + ReportHelper.generate_excel( + contractor_id, contInfo, invoices, hold_types, hold_data, + credit_note_map, gst_release_map, output_file + ) + + return output_file, None + + + + diff --git a/model/State.py b/model/State.py new file mode 100644 index 0000000..cca6d6c --- /dev/null +++ b/model/State.py @@ -0,0 +1,156 @@ +import config +import mysql.connector +from flask import request, redirect, url_for + +from model.Utilities import ResponseHandler, HtmlHelper, ItemCRUDType +from model.ItemCRUD import ItemCRUD + + +class State: + + def __init__(self): + self.isSuccess = False + self.resultMessage = "" + + # ---------------------------------------------------------- + # ADD STATE (USING ITEM CRUD) + # ---------------------------------------------------------- + def AddState(self, request): + state_name = request.form['state_Name'].strip() + + crud = ItemCRUD(ItemCRUDType.State) + + crud.AddItem( + request=request, + childname=state_name, + storedprocfetch="CheckStateExists", + storedprocadd="SaveState" + ) + + self.isSuccess = crud.isSuccess + self.resultMessage = crud.resultMessage + + return self.resultMessage + + + # ---------------------------------------------------------- + # GET ALL STATES (NO CHANGE - THIS IS CORRECT) + # ---------------------------------------------------------- + def GetAllStates(self, request): + connection = config.get_db_connection() + data = [] + + if not connection: + return [] + + try: + cursor = connection.cursor() + cursor.callproc("GetAllStates") + for res in cursor.stored_results(): + data = res.fetchall() + + self.isSuccess = True + + except mysql.connector.Error as e: + print(f"Error fetching states: {e}") + self.isSuccess = False + self.resultMessage = HtmlHelper.json_response( + ResponseHandler.fetch_failure("state"), 500 + ) + return [] + + finally: + cursor.close() + connection.close() + + return data + + + # ---------------------------------------------------------- + # CHECK STATE (USING ITEM CRUD) + # ---------------------------------------------------------- + def CheckState(self, request): + state_name = request.json.get('state_Name', '').strip() + + crud = ItemCRUD(ItemCRUDType.State) + + return crud.CheckItem( + request=request, + parentid=None, + childname=state_name, + storedprocfetch="CheckStateExists" + ) + + + # ---------------------------------------------------------- + # DELETE STATE (USING ITEM CRUD) + # ---------------------------------------------------------- + def DeleteState(self, request, id): + crud = ItemCRUD(ItemCRUDType.State) + + crud.DeleteItem( + request=request, + itemID=id, + storedprocDelete="DeleteState" + ) + + self.isSuccess = crud.isSuccess + self.resultMessage = crud.resultMessage + + return self.resultMessage + + + # ---------------------------------------------------------- + # EDIT STATE (USING ITEM CRUD) + # ---------------------------------------------------------- + def EditState(self, request, id): + state_name = request.form['state_Name'].strip() + + crud = ItemCRUD(ItemCRUDType.State) + + crud.EditItem( + request=request, + childid=id, + parentid=None, + childname=state_name, + storedprocupdate="UpdateStateById" + ) + + self.isSuccess = crud.isSuccess + self.resultMessage = crud.resultMessage + + return redirect(url_for('state.add_state')) + + + # ---------------------------------------------------------- + # GET STATE BY ID (KEEP SAME) + # ---------------------------------------------------------- + def GetStateByID(self, request, id): + connection = config.get_db_connection() + data = None + + if not connection: + return None + + try: + cursor = connection.cursor() + cursor.callproc("GetStateByID", (id,)) + for res in cursor.stored_results(): + data = res.fetchone() + + if data: + self.isSuccess = True + self.resultMessage = "Success" + else: + self.isSuccess = False + self.resultMessage = "Not Found" + + except mysql.connector.Error as e: + print(f"Error fetching state: {e}") + self.isSuccess = False + + finally: + cursor.close() + connection.close() + + return data \ No newline at end of file diff --git a/model/Subcontractor.py b/model/Subcontractor.py new file mode 100644 index 0000000..0ec659d --- /dev/null +++ b/model/Subcontractor.py @@ -0,0 +1,140 @@ +from model.Utilities import ItemCRUDType +from model.ItemCRUD import ItemCRUD + + +class Subcontractor: + def __init__(self): + self.isSuccess = False + self.resultMessage = "" + + # ---------------------------------------------------------- + # ADD + # ---------------------------------------------------------- + def AddSubcontractor(self, request): + + subcontractor = ItemCRUD(itemType=ItemCRUDType.Subcontractor) + + data = { + "Contractor_Name": request.form.get('Contractor_Name', '').strip(), + "Address": request.form.get('Address', '').strip(), + "Mobile_No": request.form.get('Mobile_No', '').strip(), + "PAN_No": request.form.get('PAN_No', '').strip(), + "Email": request.form.get('Email', '').strip(), + "Gender": request.form.get('Gender', '').strip(), + "GST_Registration_Type": request.form.get('GST_Registration_Type', '').strip(), + "GST_No": request.form.get('GST_No', '').strip(), + "Contractor_password": request.form.get('Contractor_password', '').strip() + } + + subcontractor.AddItem( + request=request, + data=data, + storedprocfetch="GetSubcontractorByName", + storedprocadd="SaveContractor" + ) + + self.isSuccess = subcontractor.isSuccess + self.resultMessage = subcontractor.resultMessage + return + + # ---------------------------------------------------------- + # GET ALL + # ---------------------------------------------------------- + def GetAllSubcontractors(self, request): + + subcontractor = ItemCRUD(itemType=ItemCRUDType.Subcontractor) + + data = subcontractor.GetAllData( + request=request, + storedproc="GetAllSubcontractors" + ) + + self.isSuccess = subcontractor.isSuccess + self.resultMessage = subcontractor.resultMessage + return data + + # ---------------------------------------------------------- + # GET BY ID + # ---------------------------------------------------------- + def GetSubcontractorByID(self, id): + + subcontractor = ItemCRUD(itemType=ItemCRUDType.Subcontractor) + + data = subcontractor.GetDataByID( + id=id, + storedproc="GetSubcontractorById" + ) + + if data: + self.isSuccess = True + else: + self.isSuccess = False + self.resultMessage = "Subcontractor not found" + + return data + + # ---------------------------------------------------------- + # CHECK (Duplicate) + # ---------------------------------------------------------- + def CheckSubcontractor(self, request): + + subcontractor = ItemCRUD(itemType=ItemCRUDType.Subcontractor) + + name = request.form.get('Contractor_Name', '').strip() + + result = subcontractor.CheckItem( + request=request, + childname=name, + storedprocfetch="GetSubcontractorByName" + ) + + self.isSuccess = subcontractor.isSuccess + self.resultMessage = subcontractor.resultMessage + return result + + # ---------------------------------------------------------- + # EDIT + # ---------------------------------------------------------- + def EditSubcontractor(self, request, subcontractor_id): + + subcontractor = ItemCRUD(itemType=ItemCRUDType.Subcontractor) + + data = { + "Contractor_Name": request.form.get('Contractor_Name', '').strip(), + "Address": request.form.get('Address', '').strip(), + "Mobile_No": request.form.get('Mobile_No', '').strip(), + "PAN_No": request.form.get('PAN_No', '').strip(), + "Email": request.form.get('Email', '').strip(), + "Gender": request.form.get('Gender', '').strip(), + "GST_Registration_Type": request.form.get('GST_Registration_Type', '').strip(), + "GST_No": request.form.get('GST_No', '').strip(), + "Contractor_password": request.form.get('Contractor_password', '').strip() + } + + subcontractor.EditItem( + request=request, + childid=subcontractor_id, + data=data, + storedprocupdate="UpdateSubcontractor" + ) + + self.isSuccess = subcontractor.isSuccess + self.resultMessage = subcontractor.resultMessage + return + + # ---------------------------------------------------------- + # DELETE + # ---------------------------------------------------------- + def DeleteSubcontractor(self, request, subcontractor_id): + + subcontractor = ItemCRUD(itemType=ItemCRUDType.Subcontractor) + + subcontractor.DeleteItem( + request=request, + itemID=subcontractor_id, + storedprocDelete="DeleteSubcontractor" + ) + + self.isSuccess = subcontractor.isSuccess + self.resultMessage = subcontractor.resultMessage + return \ No newline at end of file diff --git a/model/Utilities.py b/model/Utilities.py new file mode 100644 index 0000000..4bc4262 --- /dev/null +++ b/model/Utilities.py @@ -0,0 +1,67 @@ +from flask import flash, jsonify, json +from enum import Enum + +class ItemCRUDType(Enum): + Village = 1 + Block = 2 + District = 3 + State = 4 + HoldType = 5 + Subcontractor = 6 + GSTRelease = 7 + +class RegEx: + patternAlphabetOnly = "^[A-Za-z ]+$" + allPattern = "^(?!\s*$).+" + + +class ResponseHandler: + @staticmethod + def invalid_name(entity): + return {'status': 'error', 'message': f'Invalid {entity} name. Only letters are allowed!'} + + @staticmethod + def already_exists(entity): + return {'status': 'exists', 'message': f'{entity.capitalize()} already exists!'} + + @staticmethod + def add_success(entity): + return {'status': 'success', 'message': f'{entity.capitalize()} added successfully!'} + + @staticmethod + def add_failure(entity): + return {'status': 'error', 'message': f'Failed to add {entity}.'} + + @staticmethod + def is_available(entity): + return {'status': 'available', 'message': f'{entity.capitalize()} name is available!'} + + @staticmethod + def delete_success(entity): + return {'status': 'success', 'message': f'{entity.capitalize()} deleted successfully!'} + + @staticmethod + def delete_failure(entity): + return {'status': 'error', 'message': f'Failed to delete {entity}.'} + + @staticmethod + def update_success(entity): + return {'status': 'success', 'message': f'{entity.capitalize()} updated successfully!'} + + @staticmethod + def update_failure(entity): + return {'status': 'error', 'message': f'Failed to update {entity}.'} + + @staticmethod + def fetch_failure(entity): + return {'status': 'error', 'message': f'Failed to fetch {entity}.'} + + +class HtmlHelper: + # Helper: JSON Response Formatter + + @staticmethod + def json_response(message_obj, status_code): + return jsonify(message_obj), status_code + + diff --git a/model/Village.py b/model/Village.py new file mode 100644 index 0000000..d6f0bb2 --- /dev/null +++ b/model/Village.py @@ -0,0 +1,186 @@ + + +from model.Utilities import ResponseHandler, HtmlHelper, ItemCRUDType +import config +import mysql.connector +from model.ItemCRUD import ItemCRUD + + +class Village: + isSuccess = False + resultMessage = "" + + def __init__(self): + self.isSuccess = False + self.resultMessage = "" + self.response = {} # ✅ ADDED + self.village = ItemCRUD(itemType=ItemCRUDType.Village) + + # 🔹 Helper: sync status + def _set_status(self, village): + self.isSuccess = village.isSuccess + + # ✅ UPDATED (safe handling) + if hasattr(village, "response"): + self.response = village.response + self.resultMessage = village.response.get("message", "") + else: + self.resultMessage = village.resultMessage + + # 🔹 Helper: get request data + def _get_form_data(self, request): + block_id = request.form.get('block_Id') + village_name = request.form.get('Village_Name', '').strip() + return block_id, village_name + + def AddVillage(self, request): + block_id, village_name = self._get_form_data(request) + + if not village_name: + self.response = ResponseHandler.invalid_name("village") # ✅ UPDATED + self.resultMessage = self.response["message"] + self.isSuccess = False + return + + try: + self.village.AddItem( + request=request, + parentid=block_id, + childname=village_name, + storedprocfetch="GetVillageByNameAndBlock", + storedprocadd="SaveVillage" + ) + self._set_status(self.village) + + except Exception as e: + self.isSuccess = False + self.resultMessage = str(e) + + def GetAllVillages(self, request): + + try: + villagesdata = self.village.GetAllData( + request=request, + storedproc="GetAllVillages" + ) + self._set_status(self.village) + return villagesdata + + except Exception as e: + self.isSuccess = False + self.resultMessage = str(e) + return [] + + def CheckVillage(self, request): + block_id, village_name = self._get_form_data(request) + + if not village_name: + self.response = ResponseHandler.invalid_name("village") # ✅ UPDATED + self.resultMessage = self.response["message"] + self.isSuccess = False + return None + + try: + result = self.village.CheckItem( + request=request, + parentid=block_id, + childname=village_name, + storedprocfetch="GetVillageByNameAndBlocks" + ) + self._set_status(self.village) + return result + + except Exception as e: + self.isSuccess = False + self.resultMessage = str(e) + return None + + def DeleteVillage(self, request, village_id): + + try: + self.village.DeleteItem( + request=request, + itemID=village_id, + storedprocDelete="DeleteVillage" + ) + self._set_status(self.village) + + except Exception as e: + self.isSuccess = False + self.resultMessage = str(e) + + def EditVillage(self, request, village_id): + block_id, village_name = self._get_form_data(request) + + if not village_name: + self.response = ResponseHandler.invalid_name("village") # ✅ UPDATED + self.resultMessage = self.response["message"] + self.isSuccess = False + return + + try: + self.village.EditItem( + request=request, + childid=village_id, + parentid=block_id, + childname=village_name, + storedprocupdate="UpdateVillage" + ) + self._set_status(self.village) + + except Exception as e: + self.isSuccess = False + self.resultMessage = str(e) + + def GetVillageByID(self, id): + + try: + villagedetailsdata = self.village.GetDataByID( + id=id, + storedproc="GetVillageDetailsById" + ) + + if villagedetailsdata: + self.isSuccess = True + else: + self.isSuccess = False + self.resultMessage = "Village not found" + + return villagedetailsdata + + except Exception as e: + self.isSuccess = False + self.resultMessage = str(e) + return None + + def GetAllBlocks(self): + blocks = [] + self.isSuccess = False + self.resultMessage = "" + + connection = config.get_db_connection() + if not connection: + return [] + + try: + with connection.cursor() as cursor: + cursor.callproc('GetAllBlocks') + + for result in cursor.stored_results(): + blocks.extend(result.fetchall()) + + self.isSuccess = True + return blocks + + except mysql.connector.Error as e: + print(f"Error fetching blocks: {e}") + self.isSuccess = False + + # ✅ FIXED (removed jsonify response) + self.response = ResponseHandler.fetch_failure("block") + self.resultMessage = self.response["message"] + + return [] + + finally: + connection.close() \ No newline at end of file diff --git a/model/gst_release.py b/model/gst_release.py new file mode 100644 index 0000000..65be51e --- /dev/null +++ b/model/gst_release.py @@ -0,0 +1,160 @@ +# model/gst_release.py +from flask import request, jsonify +from model.ItemCRUD import ItemCRUD +from model.Utilities import ItemCRUDType + +class GSTRelease: + + def __init__(self): + self.isSuccess = False + self.resultMessage = "" + + # ------------------- Add GST Release ------------------- + def AddGSTRelease(self, request): + try: + gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease) + + # Print the full form data + print("===== DEBUG: FORM DATA =====") + for key, value in request.form.items(): + print(f"{key} : {value}") + print("=============================") + + data = { + "PMC_No": request.form.get("PMC_No", "").strip(), + "Invoice_No": request.form.get("Invoice_No", "").strip(), + "Basic_Amount": float(request.form.get("Basic_Amount", 0) or 0), + "Final_Amount": float(request.form.get("Final_Amount", 0) or 0), + "Total_Amount": float(request.form.get("Total_Amount", 0) or 0), + "UTR": request.form.get("UTR", "").strip(), + "Contractor_ID": int(request.form.get("Contractor_ID", 0) or 0) + } + + print("===== DEBUG: PARSED DATA =====") + print(data) + print("==============================") + + # Add GST Release + gst.AddItem( + request=request, + data=data, + storedprocfetch="CheckGSTReleaseExists", + storedprocadd="AddGSTReleaseFromExcel" + ) + + print(f"AddItem result: isSuccess={gst.isSuccess}, message={gst.resultMessage}") + + self.isSuccess = gst.isSuccess + self.resultMessage = str(gst.resultMessage) + + except Exception as e: + print("ERROR in AddGSTRelease:", e) + self.isSuccess = False + self.resultMessage = str(e) + + return jsonify({"success": self.isSuccess, "message": self.resultMessage}) + + def EditGSTRelease(self, request, gst_release_id): + try: + gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease) + + # Map form inputs to stored procedure parameters + data = { + "p_pmc_no": request.form.get("PMC_No", "").strip(), + "p_invoice_no": request.form.get("invoice_no", "").strip(), + "p_basic_amount": float(request.form.get("Basic_Amount", 0) or 0), + "p_final_amount": float(request.form.get("Final_Amount", 0) or 0), + "p_total_amount": float(request.form.get("Total_Amount", 0) or 0), + "p_utr": request.form.get("UTR", "").strip(), + "p_gst_release_id": gst_release_id + } + + print("===== DEBUG: UPDATE DATA =====") + print(data) + print("==============================") + + # Call your stored procedure + gst.EditItem( + request=request, + childid=gst_release_id, + data=data, + storedprocupdate="UpdateGSTRelease" + ) + + self.isSuccess = gst.isSuccess + self.resultMessage = str(gst.resultMessage) + + except Exception as e: + print("ERROR in EditGSTRelease:", e) + self.isSuccess = False + self.resultMessage = str(e) + + # ------------------- Delete GST Release ------------------- + def DeleteGSTRelease(self, gst_release_id): + try: + gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease) + + gst.DeleteItem( + request=None, + itemID=gst_release_id, + storedprocDelete="DeleteGSTReleaseById" + ) + + self.isSuccess = gst.isSuccess + self.resultMessage = str(gst.resultMessage) + + except Exception as e: + print("ERROR in DeleteGSTRelease:", e) + self.isSuccess = False + self.resultMessage = str(e) + + return jsonify({"success": self.isSuccess, "message": self.resultMessage}) + + # ------------------- Get All GST Releases ------------------- + def GetAllGSTReleases(self): + try: + gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease) + rows = gst.GetAllData(None, "GetAllGSTReleases") + + data = [] + for row in rows: + data.append({ + "gst_release_id": row[0], + "pmc_no": row[1], + "invoice_no": row[2], + "basic_amount": row[3], + "final_amount": row[4], + "total_amount": row[5], + "utr": row[6], + "contractor_id": row[7] + }) + + return data + + except Exception as e: + print("ERROR in GetAllGSTReleases:", e) + return [] + + # ------------------- Get GST Release By ID ------------------- + def GetGSTReleaseByID(self, gst_release_id): + try: + gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease) + row = gst.GetDataByID(gst_release_id, "GetGSTReleaseById") + + if row: + return { + "gst_release_id": row[0], + "pmc_no": row[1], + "invoice_no": row[2], + "basic_amount": row[3], + "final_amount": row[4], + "total_amount": row[5], + "utr": row[6], + "contractor_id": row[7] + } + + return None + + except Exception as e: + print("ERROR in GetGSTReleaseByID:", e) + return None \ No newline at end of file diff --git a/model/payment.py b/model/payment.py new file mode 100644 index 0000000..75f237b --- /dev/null +++ b/model/payment.py @@ -0,0 +1,236 @@ +import config +import mysql.connector +import config +import mysql.connector +from enum import Enum +from model.Utilities import ItemCRUDType + +class Paymentmodel: + + # ---------------- Database Connection ---------------- + @staticmethod + def get_connection(): + connection = config.get_db_connection() + if not connection: + return None + return connection + + # ---------------- Payment Methods ---------------- + @staticmethod + def fetch_all_payments(): + connection = Paymentmodel.get_connection() + payments = [] + if connection: + cursor = connection.cursor(dictionary=True) + try: + cursor.callproc('GetAllPayments') + for result in cursor.stored_results(): + payments = result.fetchall() + except mysql.connector.Error as e: + print(f"Error fetching payment history: {e}") + finally: + cursor.close() + connection.close() + return payments + + @staticmethod + def insert_payment(subcontractor_id, pmc_no, invoice_no, amount, tds_amount, total_amount, utr): + connection = Paymentmodel.get_connection() + if not connection: + return False + + cursor = None + try: + cursor = connection.cursor() + + cursor.callproc('GetInvoiceId', [subcontractor_id, pmc_no, invoice_no]) + + invoice_id = None + for result in cursor.stored_results(): + row = result.fetchone() + if row: + invoice_id = row[0] + + if not invoice_id: + return False + + cursor.callproc( + 'InsertPayments', + [pmc_no, invoice_no, amount, tds_amount, total_amount, utr, invoice_id] + ) + + connection.commit() + return True + + except Exception as e: + print(e) + return False + + finally: + if cursor: + cursor.close() + if connection: + connection.close() + + # @staticmethod + # def update_inpayment(subcontractor_id, pmc_no, invoice_no, amount, tds_amount, total_amount, utr): + # connection = Paymentmodel.get_connection() + # if not connection: + # return False + # try: + # cursor = connection.cursor() + # cursor.callproc('UpdateInpaymentRecord', [ + # subcontractor_id, pmc_no, invoice_no, amount, tds_amount, total_amount, utr + # ]) + # connection.commit() + # return True + # except mysql.connector.Error as e: + # print(f"Error updating inpayment: {e}") + # return False + # finally: + # cursor.close() + # connection.close() + + @staticmethod + def fetch_payment_by_id(payment_id): + connection = Paymentmodel.get_connection() + if not connection: + return None + payment_data = {} + try: + cursor = connection.cursor(dictionary=True) + cursor.callproc("GetPaymentById", (payment_id,)) + for result in cursor.stored_results(): + payment_data = result.fetchone() + if payment_data: + payment_data = [ + payment_data.get('Payment_Id'), + payment_data.get('PMC_No'), + payment_data.get('Invoice_No'), + payment_data.get('Payment_Amount'), + payment_data.get('TDS_Payment_Amount'), + payment_data.get('Total_Amount'), + payment_data.get('UTR') + ] + except mysql.connector.Error as e: + print(f"Error fetching payment data: {e}") + finally: + cursor.close() + connection.close() + return payment_data + + @staticmethod + def call_update_payment_proc(payment_id, pmc_no, invoice_no, amount, tds_amount, total_amount, utr): + connection = Paymentmodel.get_connection() + if not connection: + return False + try: + cursor = connection.cursor() + cursor.callproc("UpdatePayment", (payment_id, pmc_no, invoice_no, amount, tds_amount, total_amount, utr)) + connection.commit() + return True + except mysql.connector.Error as e: + print(f"Error updating payment: {e}") + return False + finally: + cursor.close() + connection.close() + + @staticmethod + def delete_payment(payment_id): + connection = Paymentmodel.get_connection() + if not connection: + return False, None, None + try: + cursor = connection.cursor(dictionary=True) + # Fetch PMC & Invoice before deleting + cursor.callproc('GetPaymentPMCInvoiceById', [payment_id]) + record = {} + for result in cursor.stored_results(): + record = result.fetchone() or {} + if not record: + return False, None, None + pmc_no = record['PMC_No'] + invoice_no = record['Invoice_No'] + # Delete payment + cursor.callproc("DeletePayment", (payment_id,)) + connection.commit() + # Reset inpayment fields + # cursor.callproc("ResetInpayment", [pmc_no, invoice_no]) + # connection.commit() + return True, pmc_no, invoice_no + except mysql.connector.Error as e: + print(f"Error deleting payment: {e}") + return False, None, None + finally: + cursor.close() + connection.close() + + # ---------------- Item CRUD Methods ---------------- + @staticmethod + def fetch_items(item_type: ItemCRUDType): + connection = Paymentmodel.get_connection() + items = [] + if connection: + cursor = connection.cursor(dictionary=True) + try: + cursor.callproc('GetItemsByType', [item_type.value]) + for result in cursor.stored_results(): + items = result.fetchall() + except mysql.connector.Error as e: + print(f"Error fetching {item_type.name}: {e}") + finally: + cursor.close() + connection.close() + return items + + @staticmethod + def insert_item(item_type: ItemCRUDType, name: str): + connection = Paymentmodel.get_connection() + if not connection: + return False + try: + cursor = connection.cursor() + cursor.callproc('InsertItem', [item_type.value, name]) + connection.commit() + return True + except mysql.connector.Error as e: + print(f"Error inserting {item_type.name}: {e}") + return False + finally: + cursor.close() + connection.close() + + @staticmethod + def update_item(item_type: ItemCRUDType, item_id: int, new_name: str): + connection = Paymentmodel.get_connection() + if not connection: + return False + try: + cursor = connection.cursor() + cursor.callproc('UpdateItem', [item_type.value, item_id, new_name]) + connection.commit() + return True + except mysql.connector.Error as e: + print(f"Error updating {item_type.name}: {e}") + return False + finally: + cursor.close() + connection.close() + + @staticmethod + def delete_item(item_type: ItemCRUDType, item_id: int): + connection = Paymentmodel.get_connection() + if not connection: + return False + try: + cursor = connection.cursor() + cursor.callproc('DeleteItem', [item_type.value, item_id]) + connection.commit() + return True + except mysql.connector.Error as e: + print(f"Error deleting {item_type.name}: {e}") + return False + finally: + cursor.close() + connection.close() \ No newline at end of file diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..61475b0 --- /dev/null +++ b/requirements.txt @@ -0,0 +1,4 @@ +Flask +mysql-connector-python +openpyxl +pandas diff --git a/static/css/base.css b/static/css/base.css new file mode 100644 index 0000000..9094baf --- /dev/null +++ b/static/css/base.css @@ -0,0 +1,89 @@ +body { + margin: 0; + font-family: Arial, sans-serif; + background-color: #f5f5f5; + display: flex; + } + + /* Sidebar */ + .sidebar { + width: 250px; + height: 100vh; + background-color: #ffffff; + position: fixed; + left: -250px; + transition: left 0.3s ease; + box-shadow: 2px 0 5px rgba(0, 0, 0, 0.1); + padding-top: 20px; + } + .sidebar.open { + left: 0; + } + + /* Sidebar Navigation */ + .nav-menu { + list-style: none; + padding: 0; + margin: 0; + } + + .nav-item { + width: 100%; + } + + .nav-link { + display: flex; + align-items: center; + text-decoration: none; + color: #333; + padding: 15px 20px; + font-size: 16px; + transition: background-color 0.3s, color 0.3s; + } + + .nav-link i { + margin-right: 10px; + font-size: 18px; + } + + .nav-link:hover, .nav-link.active { + background-color: #e6f7ff; + color: #007bff; + } + + /* Menu Button */ + .menu-icon { + position: absolute; + top: 15px; + left: 15px; + font-size: 24px; + cursor: pointer; + background: none; + border: none; + } + + /* Content Area */ + .content { + margin-left: 0; + padding: 20px; + width: 100%; + transition: margin-left 0.3s ease; + } + .content.shift { + margin-left: 250px; + } +@media (max-width: 768px) { + .sidebar { + width: 200px; + left: -200px; + } + + .sidebar.open { + left: 0; + } + + .content.shift { + margin-left: 200px; + } +} + diff --git a/static/css/index.css b/static/css/index.css new file mode 100644 index 0000000..1c22a93 --- /dev/null +++ b/static/css/index.css @@ -0,0 +1,226 @@ +body { + margin: 0; + font-family: Arial, sans-serif; + display: flex; + flex-direction: column; + background-color: #f5f5f5; + } + + /* Sidebar */ + .sidebar { + width: 250px; + height: 100%; + background-color: #ffffff; + box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); + display: flex; + flex-direction: column; + align-items: center; + padding-top: 20px; + position: fixed; + } + .logo { + width: 80px; + margin-bottom: 20px; + } + .nav-menu { + width: 100%; + list-style: none; + padding: 0; + margin: 0; + } + .nav-item { + width: 100%; + } + .nav-link { + display: flex; + align-items: center; + text-decoration: none; + color: #333; + padding: 15px 20px; + font-size: 16px; + transition: background-color 0.3s, color 0.3s; + } + .nav-link:hover, + .nav-link.active { + background-color: #e6f7ff; + color: #007bff; + } + .nav-link i { + margin-right: 10px; + font-size: 18px; + } + .user-section { + margin-top: auto; + width: 100%; + padding: 20px; + display: flex; + align-items: center; + border-top: 1px solid #eee; + } + .user-section img { + width: 50px; + height: 50px; + border-radius: 50%; + margin-right: 10px; + } + .user-info { + display: flex; + flex-direction: column; + } + .user-info span { + font-size: 14px; + color: #333; + } + + + /* Main content area */ + .content { + margin-left: 250px; + padding: 20px; + width: calc(100% - 250px); + } + + /* Menu Cards */ + .menu { + display: flex; + flex-wrap: wrap; + gap: 20px; + } + .card { + background-color: #fff; + box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); + border-radius: 8px; + padding: 20px; + text-align: center; + flex: 1 1 calc(30% - 20px); + max-width: calc(30% - 20px); + + } + .card h2 { + margin: 0 0 10px; + font-size: 20px; + } + .btn { + display: inline-block; + padding: 10px 20px; + color: #fff; + background-color: #007bff; + border-radius: 4px; + text-decoration: none; + font-size: 14px; + transition: background-color 0.3s; + } + .btn:hover { + background-color: #0056b3; + } + +/* Company Info */ +.company-info { + text-align: center; + padding: 20px; + background-color: Whitesmoke; + color: blue; + margin-left: 250px; +} + +.company-name { + font-size: 30px; + font-weight: bold; + margin: 0; +} + +.app-name { + font-weight: bold; + app-name-shadow:0 2px 4px rgba(0, 0, 0, 0.1); + font-size: 24px; + margin: 5px 0 0; +} + + + + /* Responsive Design */ +@media screen and (max-width: 768px) { + .sidebar { + width: 100%; + height: auto; + position: static; + } + .content { + margin-left: 0; + width: 100%; + } + .menu { + flex-direction: column; + align-items: center; + } + .card { + flex: 1 1 100%; + max-width: 100%; + min-width: 50%; + } +} + +@media screen and (max-width: 480px) { + .nav-link { + font-size: 14px; + padding: 10px 15px; + } + .btn { + font-size: 12px; + padding: 8px 15px; + } +} + +@media screen and (max-width: 768px) { + .sidebar { + width: 100%; + height: auto; + position: static; + } + + .content { + margin-left: 0; + width: 100%; + } + + .menu { + flex-direction: column; + align-items: center; + } + + .card { + flex: 1 1 100%; + max-width: 100%; + min-width: 50%; + } + + .company-info { + margin-left: 0; + } +} + +@media screen and (max-width: 480px) { + .nav-link { + font-size: 14px; + padding: 10px 15px; + } + + .btn { + font-size: 12px; + padding: 8px 15px; + } + + .user-section { + flex-direction: column; + align-items: flex-start; + gap: 10px; + } + + .user-section img { + margin-right: 0; + } + + .card { + min-width: 90%; + } +} diff --git a/static/css/invoice.css b/static/css/invoice.css new file mode 100644 index 0000000..ecf18e0 --- /dev/null +++ b/static/css/invoice.css @@ -0,0 +1,395 @@ +/* General Styles */ + +h2 { + text-align: center; + font-size: 24px; + color: #333; + margin-bottom: 20px; +} + +/* Form Styling */ +form { + width:50%; + background: #fff; + padding: 20px; + border-radius: 8px; + box-shadow: 0px 4px 10px rgba(0, 0, 0, 0.1); + display: flex; + flex-direction: column; + gap: 15px; +} + +/* Responsive Form Layout */ +.row1, +.row2, +.row3 { + display: grid; + gap: 15px; +} + +.row2, +.row3 { + grid-template-columns: repeat(auto-fit, minmax(250px, 1fr)); +} + +label { + font-weight: bold; + display: block; + margin-bottom: 6px; +} + +input, +textarea, +select { + width: 100%; + padding: 10px; + border: 1px solid #ccc; + border-radius: 6px; + font-size: 16px; + box-sizing: border-box; +} + +/* Button Styling */ +.button { + padding: 12px; + background-color: #007bff; + color: white; + border: none; + border-radius: 6px; + font-size: 18px; + cursor: pointer; + transition: background-color 0.3s ease-in-out; +} + +.button:hover { + background-color: #0056b3; +} + +/* Dynamic Hold Amount Fields */ +.hold-row { + display: flex; + flex-wrap: wrap; + gap: 10px; +} + +.hold-amount-field { + display: flex; + align-items: center; + gap: 10px; + padding: 10px; + background-color: #f9f9f9; + border: 1px solid #ddd; + border-radius: 4px; + flex-wrap: wrap; +} + +.hold-amount-field select, +.hold-amount-field input { + flex: 1; + min-width: 100px; +} + +.hold-amount-field button { + background-color: red; + color: white; + font-size: 14px; + padding: 8px 10px; + border: none; + border-radius: 4px; + cursor: pointer; +} + +.hold-amount-field button:hover { + background-color: #c0392b; +} + +/* Success Alert Box */ +.success-alert { + position: fixed; + top: 20px; + left: 50%; + transform: translateX(-50%); + background-color: #4CAF50; + color: #fff; + padding: 15px 25px; + border-radius: 5px; + box-shadow: 0px 4px 10px rgba(0, 0, 0, 0.1); + opacity: 0; + visibility: hidden; + transition: opacity 0.5s ease, visibility 0.5s ease; + z-index: 1000; +} + +.success-alert.show { + opacity: 1; + visibility: visible; +} + +/* Table Styles */ +.invoice-table { + width: 100%; + border-collapse: collapse; + margin-top: 20px; + overflow-x: auto; +} + +.invoice-table th, +.invoice-table td { + border: 1px solid #ddd; + padding: 10px; + text-align: left; + font-size: 14px; + word-break: break-word; +} + +.invoice-table th { + background-color: #007bff; + color: #fff; +} + +.invoice-table tr:nth-child(even) { + background-color: #f9f9f9; +} + +.invoice-table tr:hover { + background-color: #f1f1f1; +} + + + +/* icon */ + .icon { + width: 20px; + height: 20px; + cursor: pointer; + transition: transform 0.2s ease-in-out, opacity 0.2s ease-in-out; + } + + .icon:hover { + transform: scale(1.5); + opacity: 0.8; + } + + .edit-icon:hover { + filter: drop-shadow(0 0 5px #007bff); /* Blue glow for edit */ + } + + .delete-icon:hover { + filter: drop-shadow(0 0 5px #ff0000); /* Red glow for delete */ + } + + +/* Center the buttons and apply consistent styling */ +.button-container { + display: flex; + justify-content: center; /* Center buttons horizontally */ + gap: 10px; /* Space between buttons */ + margin-top: 20px; /* Add some top margin */ +} + +.action-button { + background-color: #007BFF; /* Blue background */ + color: white; /* White text */ + padding: 15px 30px; /* Larger padding for bigger buttons */ + border: none; /* Remove border */ + border-radius: 5px; /* Rounded corners */ + cursor: pointer; /* Pointer cursor on hover */ + font-size: 18px; /* Larger font size */ + text-align: center; /* Center text */ + text-decoration: none; /* Remove underline */ + transition: background-color 0.3s ease; /* Smooth hover transition */ +} + +.action-button:hover { + background-color: #0056b3; /* Darker blue on hover */ +} + +#addStateForm, #stateTable { + display: none; /* Initially hide both the form and the table */ +} + +/* Success Popup */ +.success-popup { + display: none; + color: green; + font-size: 1.2em; + margin-top: 10px; +} + +/* Sorting buttons */ +.sortable .sort-buttons { + margin-left: 5px; +} + + +.sortable { + cursor: pointer; + user-select: none; + } +.sort-buttons a { + text-decoration: none; + color: black; + font-weight: bold; + margin-left: 5px; + margin-right: 5px; +} +.sort-buttons a:hover { + text-decoration: underline; + } + .back-button { + display: inline-block; + margin-top: 20px; + padding: 8px 15px; + background-color: #28a745; + color: white; + border-radius: 5px; + text-decoration: none; + font-size: 16px; + } + .back-button:hover { + background-color: #218838; + } + + + + + +span .sort-desc:hover{ +cursor: pointer; +} + +span .sort-asc:hover{ +cursor: pointer; +} + + +/* Responsive Design */ +@media (max-width: 1024px) { + form { + padding: 15px; + } + + .row2, + .row3 { + grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); + } + + .invoice-table th, + .invoice-table td { + font-size: 12px; + padding: 8px; + } +} + +@media (max-width: 768px) { + .row1, + .row2, + .row3 { + grid-template-columns: 1fr; + } + + .hold-amount-field { + flex-direction: column; + gap: 5px; + align-items: flex-start; + } + + .hold-amount-field select, + .hold-amount-field input { + width: 100%; + } + + .invoice-table { + display: block; + overflow-x: auto; + white-space: nowrap; + } +} + +@media (max-width: 480px) { + body { + max-width: 100%; + padding: 10px; + } + + form { + padding: 10px; + } + + .invoice-table th, + .invoice-table td { + font-size: 12px; + padding: 5px; + } + + button { + font-size: 16px; + padding: 10px; + } +} +/* Additional Media Queries */ + +/* For tablets and medium devices */ +@media (max-width: 992px) { + form { + width: 90%; + } + + .button-container { + flex-direction: column; + align-items: center; + } + + .action-button { + width: 100%; + max-width: 300px; + } +} + +/* For smaller tablets and large phones */ +@media (max-width: 600px) { + .button-container { + flex-direction: column; + align-items: stretch; + } + + .action-button { + width: 100%; + } + + .icon { + width: 16px; + height: 16px; + } + + .success-alert { + width: 90%; + font-size: 14px; + padding: 10px 15px; + } +} + +/* For very small phones */ +@media (max-width: 360px) { + h2 { + font-size: 20px; + } + + .back-button { + width: 100%; + text-align: center; + font-size: 14px; + padding: 8px 10px; + } + + .sort-buttons a { + font-size: 12px; + } + + .invoice-table th, + .invoice-table td { + font-size: 10px; + padding: 4px; + } +} + diff --git a/static/css/invoice1.css b/static/css/invoice1.css new file mode 100644 index 0000000..11d9e52 --- /dev/null +++ b/static/css/invoice1.css @@ -0,0 +1,313 @@ +/* General Styling */ +body { + font-family: Arial, sans-serif; + margin: 20px; + background-color: #f9f9f9; + color: #333; +} + +/* Header Styling */ +h1, h2, h3 { + color: #0056b3; + text-align: center; +} + +/* Table Styling */ +table { + width: 100%; + border-collapse: collapse; + margin: 20px 0; +} + +table th, table td { + border: 1px solid #ddd; + padding: 8px; + text-align: center; +} + +table th { + background-color: #007bff; + color: white; + font-weight: bold; + cursor: pointer; + position: relative; +} + +table tr:nth-child(even) { + background-color: #f2f2f2; +} + +table tr:hover { + background-color: #ddd; +} + +/* Sort Dropdown */ +.sort-options { + display: none; + position: absolute; + background: white; + border: 1px solid black; + padding: 5px; + z-index: 100; + width: 120px; + text-align: center; +} + +.sort-options button { + display: block; + width: 100%; + border: none; + background: none; + padding: 5px; + cursor: pointer; +} + +.sort-options button:hover { + background-color: #f0f0f0; +} + +/* Form Styling */ +form { + max-width: 700px; + margin: 0 auto; + padding: 20px; + background: #fff; + border: 1px solid #ddd; + border-radius: 5px; + box-shadow: 0 0 10px rgba(0, 0, 0, 0.1); +} + +form label { + display: block; + margin-bottom: 5px; + font-weight: bold; +} + +form input[type="text"], +form input[type="number"], +form input[type="date"], +form input[type="email"], +form textarea, +select { + width: 100%; + padding: 10px; + margin-bottom: 15px; + border: 1px solid #ddd; + border-radius: 5px; + box-sizing: border-box; +} + +form textarea { + resize: vertical; +} + +/* Button Styling */ +.button, form button { + width: 100%; + padding: 10px; + background-color: #007bff; + color: white; + border: none; + border-radius: 5px; + cursor: pointer; + font-size: 16px; + text-align: center; +} + +.button:hover, form button:hover { + background-color: #0056b3; +} + +/* Back Button */ +.back-button { + background-color: red; + color: white; + padding: 10px 20px; + text-decoration: none; + font-weight: bold; + border-radius: 5px; + display: inline-block; + margin-top: 20px; + text-align: center; +} + +.back-button:hover { + background-color: darkred; +} + +/* Success Alert */ +.success-alert { + display: none; + position: fixed; + top: 50%; + left: 50%; + transform: translate(-50%, -50%); + background-color: #d4edda; + color: #155724; + padding: 20px; + border-radius: 10px; + box-shadow: 0 0 10px rgba(0, 0, 0, 0.2); + text-align: center; + font-size: 18px; + font-weight: bold; +} + +.success-popup i { + color: green; + font-size: 30px; + margin-right: 10px; +} + +/* Error Message */ +.error-message { + color: red; + font-size: 14px; + margin-top: 5px; +} + +/* Table Icons */ +td img { + width: 20px; /* Adjust as needed */ + height: 20px; /* Adjust as needed */ + transition: transform 0.3s ease; /* Smooth transition for hover effect */ +} + +td img:hover { + transform: scale(1.2); /* Slight zoom effect on hover */ +} + +/* Select Dropdown */ +select { + width: 100%; + padding: 12px 15px; + margin: 10px 0; + border: 1px solid #ccc; + border-radius: 5px; + font-size: 14px; + background-color: #fff; + box-sizing: border-box; + appearance: none; +} + +/* Custom Dropdown Arrow */ +select::-ms-expand { + display: none; +} + +select:focus { + outline: none; + border-color: #4CAF50; +} + +select option { + padding: 12px 15px; + font-size: 14px; +} + +option[disabled] { + color: #aaa; + font-style: italic; +} + +/* Save Button */ +.save-button { + background-color: green; + color: white; + padding: 15px; + text-decoration: none; + font-weight: bold; + border-radius: 5px; + display: flex; + justify-content: center; + align-items: center; + margin: 20px auto; + text-align: center; +} + +.save-button:hover { + background-color: darkgreen; +} + +/* Responsive Design */ +@media (max-width: 768px) { + form { + max-width: 100%; + padding: 15px; + } + + table th, table td { + padding: 6px; + font-size: 14px; + } + + .button, form button { + font-size: 14px; + padding: 8px; + } +} +/* Additional Media Queries */ + +/* For tablets and medium-sized screens */ +@media (max-width: 992px) { + form { + padding: 18px; + } + + table th, table td { + font-size: 14px; + padding: 6px; + } + + .save-button, .back-button, .button, form button { + font-size: 15px; + } +} + +/* For smaller tablets and large phones */ +@media (max-width: 600px) { + body { + margin: 10px; + } + + h1, h2, h3 { + font-size: 20px; + } + + table th, table td { + font-size: 13px; + padding: 5px; + } + + .success-alert { + width: 90%; + font-size: 16px; + padding: 15px; + } + + .button, form button, .save-button, .back-button { + padding: 10px; + font-size: 14px; + } +} + +/* For very small phones */ +@media (max-width: 360px) { + h1, h2, h3 { + font-size: 18px; + } + + .save-button, .back-button { + width: 100%; + font-size: 13px; + padding: 10px; + } + + table th, table td { + font-size: 12px; + padding: 4px; + } + + form input, form select, form textarea { + font-size: 14px; + } +} diff --git a/static/css/report.css b/static/css/report.css new file mode 100644 index 0000000..686905d --- /dev/null +++ b/static/css/report.css @@ -0,0 +1,181 @@ +h2 { + text-align: center; + font-size: 24px; + color: #333; + margin-bottom: 20px; +} + +/* Form Styling */ +.info { + width: 60%; + background: #fff; + padding: 20px; + border-radius: 8px; + box-shadow: 0px 4px 10px rgba(0, 0, 0, 0.1); + display: flex; + flex-direction: column; + gap: 15px; +} + +/* Responsive Form Layout */ +.row1, +.row2, +.row3 { + display: grid; + gap: 15px; +} + +.row2, +.row3 { + grid-template-columns: repeat(auto-fit, minmax(250px, 1fr)); +} + +label { + font-weight: bold; + display: block; + margin-bottom: 6px; +} + +input, +textarea, +select { + width: 100%; + padding: 10px; + border: 1px solid #ccc; + border-radius: 6px; + font-size: 16px; + box-sizing: border-box; +} + +/* Table styling */ +.table-wrapper { + overflow-x: auto; +} + +table { + width: 100%; + border-collapse: collapse; + margin: 20px 0; +} + +table th, table td { + border: 1px solid #ddd; + padding: 8px; + text-align: center; +} + +table th { + background-color: #007bff; + color: white; + font-weight: bold; +} + +table tr:nth-child(even) { + background-color: #f2f2f2; +} + +table tr:hover { + background-color: #ddd; +} + +.download-btn { + display: inline-block; + padding: 10px 15px; + background-color: #28a745; + color: white; + text-decoration: none; + border-radius: 5px; + margin-top: 15px; +} + +.total-table { + width: 35%; +} + +/* Responsive Design */ +@media (max-width: 1024px) { + .info { + padding: 15px; + } + + .row2, + .row3 { + grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); + } + + .invoice-table th, + .invoice-table td { + font-size: 12px; + padding: 8px; + } +} + +@media (max-width: 768px) { + .row1, + .row2, + .row3 { + grid-template-columns: 1fr; + } +} + +@media (max-width: 480px) { + body { + max-width: 100%; + padding: 10px; + } + + .info { + padding: 10px; + } + + table th, + table td { + font-size: 10px; + padding: 6px; + } +} + +/* Ensure table responsiveness */ +@media (max-width: 600px) { + .table-wrapper { + overflow-x: auto; + } + + table { + min-width: 600px; + } +} +/* Extra Small Devices (max-width: 360px) */ +@media (max-width: 360px) { + h2 { + font-size: 20px; + margin-bottom: 15px; + } + + .info { + width: 100%; + padding: 8px; + } + + input, + textarea, + select { + font-size: 14px; + padding: 8px; + } + + .download-btn { + font-size: 14px; + padding: 8px 12px; + } + + table th, + table td { + font-size: 9px; + padding: 5px; + } + + .total-table { + width: 100%; + } +} diff --git a/static/css/show_excel.css b/static/css/show_excel.css new file mode 100644 index 0000000..9563d2a --- /dev/null +++ b/static/css/show_excel.css @@ -0,0 +1,212 @@ + /* General Styles */ +body { + font-family: Arial, sans-serif; + margin: 20px; + padding: 20px; + background-color: #f4f4f9; + color: #333; +} + +/* Headings */ +h1, h2 { + color: #2c3e50; +} + +/* File Information Section */ +ul { + list-style-type: none; + padding: 0; +} + +ul li { + background: #ecf0f1; + margin: 5px 0; + padding: 10px; + border-radius: 5px; +} + +/* Error Messages */ +.errors { + background-color: #ffdddd; + border: 1px solid #ff4d4d; + padding: 15px; + margin-bottom: 20px; + border-radius: 5px; +} + +.errors h2 { + color: #ff4d4d; +} + +.error { + color: #d8000c; + font-weight: bold; +} + +/* Table Styles */ +.table-container { + max-width: 100%; + overflow-x: auto; /* Allows horizontal scrolling */ + margin-bottom: 20px; + border: 1px solid #ddd; /* Optional for better visibility */ +} + +table { + width: 100%; + border-collapse: collapse; + background-color: white; + table-layout: auto; /* Let the columns adjust based on content */ +} + +th, td { + padding: 10px; + border: 1px solid #ddd; + text-align: left; + word-wrap: break-word; /* Prevents text from overflowing */ +} + +th { + background-color: #3498db; + color: white; +} + +tr:nth-child(even) { + background-color: #f2f2f2; +} + +/* Set input width to 100px */ +input[type="text"] { + width: 200px; /* Set a fixed width of 100px */ + padding: 8px; + border: 1px solid #ccc; + border-radius: 4px; + box-sizing: border-box; /* Ensures padding is included in width calculation */ + white-space: nowrap; /* Prevent line breaks */ + overflow: hidden; /* Prevents overflow */ + text-overflow: ellipsis; /* Shows ellipsis if the content is too long */ +} + +/* Buttons */ +button { + display: inline-block; + background-color: #27ae60; + color: white; + padding: 10px 15px; + text-decoration: none; + border: none; + cursor: pointer; + border-radius: 5px; + transition: background 0.3s ease; + font-size: 16px; + width: 100%; /* Ensures button is centered */ +} + +.back-button { + display: inline-block; + background-color: #27ae60; + color: white; + padding: 10px 15px; + text-decoration: none; + border: none; + cursor: pointer; + border-radius: 5px; + transition: background 0.3s ease; + font-size: 16px; + width: 10%; /* Ensures button is centered */ +} + +/* Hover Effects */ +button:hover, .back-button:hover { + background-color: #219150; +} + +button:disabled { + background-color: #95a5a6; + cursor: not-allowed; +} + +/* Back Button */ +.back-button { + background-color: #e74c3c; +} + +.back-button:hover { + background-color: #c0392b; +} + +/* Center Save Data Button */ +form { + display: flex; + flex-direction: column; + align-items: center; +} + +.save-button { + margin-top: 20px; + width: 50%; +} + +/* Responsive Design */ +@media (max-width: 768px) { + .table-container { + display: block; + overflow-x: auto; + white-space: nowrap; + } + + table { + width: 100%; + table-layout: auto; /* Let the table adjust to fit its content */ + } + + input[type="text"] { + width: 100px; /* Fixed width of 100px */ + padding: 6px; /* Smaller padding on mobile */ + } + + button, .back-button { + width: 100%; + text-align: center; + } +} +@media (max-width: 480px) { + body { + margin: 10px; + padding: 10px; + } + + h1, h2 { + font-size: 20px; + text-align: center; + } + + input[type="text"] { + width: 80px; + font-size: 14px; + } + + th, td { + font-size: 12px; + padding: 6px; + } + + .save-button { + width: 100%; + } + + .back-button { + width: 100%; + font-size: 14px; + } + + button { + font-size: 14px; + padding: 8px 12px; + } + + .errors { + font-size: 14px; + padding: 10px; + } +} + diff --git a/static/css/style.css b/static/css/style.css new file mode 100644 index 0000000..1e4ea1d --- /dev/null +++ b/static/css/style.css @@ -0,0 +1,448 @@ + /* General styling */ +body { + font-family: Arial, sans-serif; + margin: 20px; + background-color: #f9f9f9; + color: #333; +} + +/* Header styling */ +h1 { + color: back; + text-align: center; +} +h2 { + color: #0056b3; + text-align: center; +} +h3 { + color: #0056b3; +} + +/* Table styling */ +table { + width: 100%; + border-collapse: collapse; + margin: 20px 0; +} + +table th, table td { + border: 1px solid #ddd; + padding: 8px; + text-align: center; +} + +table th { + background-color: #007bff; + color: white; + font-weight: bold; +} + +table tr:nth-child(even) { + background-color: #f2f2f2; +} + +table tr:hover { + background-color: #ddd; +} + + + +/* Form styling */ +form { + max-width: 600px; + margin: 0 auto; + padding: 20px; + background: #fff; + border: 1px solid #ddd; + border-radius: 5px; + box-shadow: 0 0 10px rgba(0, 0, 0, 0.1); +} + +form label { + display: block; + margin-bottom: 5px; + font-weight: bold; +} + + + +form input[type="text"], form input[type="number"], form input[type="date"],form input[type="email"], form textarea { + width: 100%; + padding: 10px; + margin-bottom: 15px; + border: 1px solid #ddd; + border-radius: 5px; + box-sizing: border-box; +} + +form textarea { + resize: vertical; +} + + +form button { + width: 100%; + padding: 10px; + background-color: #007bff; + color: white; + border: none; + border-radius: 5px; + cursor: pointer; + font-size: 16px; +} + +form button:hover { + background-color: #0056b3; +} + +/* Style the

    elements to display inline */ +h1 { + display: inline-block; + margin-right: 10px; /* Spacing between buttons */ +} + +/* Style the tags to look like buttons */ +.button { + text-decoration: none; /* Remove underline */ + padding: 10px 20px; /* Add padding for size */ + background-color: #4CAF50; /* Green background color */ + color: white; /* White text */ + font-size: 16px; /* Font size */ + border-radius: 5px; /* Rounded corners */ + border: none; /* Remove default border */ + cursor: pointer; /* Cursor style */ + transition: background-color 0.3s ease; /* Smooth transition for hover */ +} + +/* Change background color on hover */ +.button:hover { + background-color: #45a049; +} + +/* Style for the Back Button */ +.back-button { + background-color: red; + color: white; + padding: 10px 20px; + text-decoration: none; + font-weight: bold; + border-radius: 5px; + display: inline-block; + margin-top: 20px; + text-align: center; +} + +.back-button:hover { + background-color: darkred; +} + +/* Styling for the select dropdown */ +select { + width: 100%; /* Use full width for better responsiveness */ + padding: 12px 15px; /* Increased padding for better spacing */ + margin: 10px 0; /* Spacing above and below */ + border: 1px solid #ccc; /* Border color */ + border-radius: 5px; /* Rounded corners */ + font-size: 14px; /* Font size for text */ + background-color: #fff; /* White background */ + box-sizing: border-box; /* Ensures padding and border are included in width */ + appearance: none; /* Remove default dropdown arrow for custom styling */ +} + +/* Custom dropdown arrow */ +select::-ms-expand { + display: none; /* Remove the default arrow for IE/Edge */ +} + +select:focus { + outline: none; /* Remove outline when focused */ + border-color: #4CAF50; /* Change border color when focused */ +} + +/* Option styling for the select */ +select option { + padding: 12px 15px; /* Padding for each option */ + font-size: 14px; /* Font size */ +} + +/* Style the placeholder (default) option */ +option[disabled] { + color: #aaa; /* Light grey color for disabled option */ + font-style: italic; /* Italicize the disabled option */ +} + +/* Label styling */ +label { + font-size: 16px; + font-weight: bold; + margin-bottom: 5px; + display: block; +} + + + + +.save-button { + background-color: Green; + color: white; + padding: 15px; + text-decoration: none; + font-weight: bold; + border-radius: 5px; + display:flex; + justify-content: center; + align-item: center; + margin: 20px; + text-align: center; + +} + +.save-button:hover { + background-color: darkGreen; +} + + +.success-popup { + display: none; + position: fixed; + top: 50%; + left: 50%; + transform: translate(-50%, -50%); + background-color: #d4edda; + color: #155724; + padding: 20px; + border-radius: 10px; + box-shadow: 0 0 10px rgba(0, 0, 0, 0.2); + text-align: center; + font-size: 18px; + font-weight: bold; +} +.success-popup i { + color: green; + font-size: 30px; + margin-right: 10px; +} + + .error-message { + color: red; + font-size: 14px; + margin-top: 5px; + } + + + .icon { + width: 20px; + height: 20px; + cursor: pointer; + transition: transform 0.2s ease-in-out, opacity 0.2s ease-in-out; + } + + .icon:hover { + transform: scale(1.5); + opacity: 0.8; + } + + .edit-icon:hover { + filter: drop-shadow(0 0 5px #007bff); /* Blue glow for edit */ + } + + .delete-icon:hover { + filter: drop-shadow(0 0 5px #ff0000); /* Red glow for delete */ + } + + /* Search Bar Container */ +.search-container { + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 10px; +} + +/* Search Bar Input */ +#searchBar { + padding: 8px; + font-size: 14px; + border: 1px solid #ccc; + border-radius: 5px; + width: 200px; + transition: 0.3s; +} + +/* Search Bar Focus Effect */ +#searchBar:focus { + border-color: #007bff; + outline: none; + box-shadow: 0 0 5px rgba(0, 123, 255, 0.5); +} + + + + + +/* Center the buttons and apply consistent styling */ +.button-container { + display: flex; + justify-content: center; /* Center buttons horizontally */ + gap: 10px; /* Space between buttons */ + margin-top: 20px; /* Add some top margin */ +} + +.action-button { + background-color: #007BFF; /* Blue background */ + color: white; /* White text */ + padding: 15px 30px; /* Larger padding for bigger buttons */ + border: none; /* Remove border */ + border-radius: 5px; /* Rounded corners */ + cursor: pointer; /* Pointer cursor on hover */ + font-size: 18px; /* Larger font size */ + text-align: center; /* Center text */ + text-decoration: none; /* Remove underline */ + transition: background-color 0.3s ease; /* Smooth hover transition */ +} + +.action-button:hover { + background-color: #0056b3; /* Darker blue on hover */ +} + +#addStateForm, #stateTable { + display: none; /* Initially hide both the form and the table */ +} + +/* Success Popup */ +.success-popup { + display: none; + color: green; + font-size: 1.2em; + margin-top: 10px; +} + +/* Sorting buttons */ +.sortable .sort-buttons { + margin-left: 5px; +} + + +.sortable { + cursor: pointer; + user-select: none; + } +.sort-buttons a { + text-decoration: none; + color: black; + font-weight: bold; + margin-left: 5px; + margin-right: 5px; +} +.sort-buttons a:hover { + text-decoration: underline; + } + .back-button { + display: inline-block; + margin-top: 20px; + padding: 8px 15px; + background-color: #28a745; + color: white; + border-radius: 5px; + text-decoration: none; + font-size: 16px; + } + .back-button:hover { + background-color: #218838; + } + + + + +span .sort-desc:hover{ +cursor: pointer; +} + +span .sort-asc:hover{ +cursor: pointer; +} +.sortable select { + background-color: #007BFF; /* Blue background */ + color: white; /* White text */ + border: none; + border-radius: 4px; + padding: 5px 10px; + font-size: 14px; + cursor: pointer; + appearance: none; /* Remove default browser styling */ + background-image: url("data:image/svg+xml;charset=US-ASCII,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' fill='white' class='bi bi-caret-down-fill' viewBox='0 0 16 16'%3E%3Cpath d='M7.247 11.14 2.451 5.658c-.566-.63-.106-1.658.753-1.658h9.592c.86 0 1.32 1.027.753 1.658L8.753 11.14a1 1 0 0 1-1.506 0z'/%3E%3C/svg%3E"); + background-repeat: no-repeat; + background-position: right 10px center; + background-size: 12px; +} + +.sortable select:focus { + outline: none; + background-color: #0056b3; /* Darker blue on focus */ +} +.download-btn { + display: inline-block; + padding: 10px 15px; + background-color: #28a745; + color: white; + text-decoration: none; + border-radius: 5px; + margin-top: 15px; + } +@media (max-width: 480px) { + body { + margin: 10px; + padding: 10px; + } + + h1, h2, h3 { + font-size: 18px; + text-align: center; + } + + table th, table td { + font-size: 12px; + padding: 6px; + } + + form { + padding: 10px; + max-width: 100%; + } + + form input[type="text"], + form input[type="number"], + form input[type="date"], + form input[type="email"], + form textarea, + select { + padding: 8px; + font-size: 14px; + } + + form button, + .button, + .back-button, + .save-button, + .action-button { + font-size: 14px; + padding: 10px; + width: 100%; + } + + .button-container { + flex-direction: column; + gap: 10px; + } + + #searchBar { + width: 100%; + font-size: 14px; + } + + .sortable select { + font-size: 14px; + padding: 8px 10px; + background-position: right 8px center; + } +} diff --git a/static/css/subcontractor_report.css b/static/css/subcontractor_report.css new file mode 100644 index 0000000..b7ab1ff --- /dev/null +++ b/static/css/subcontractor_report.css @@ -0,0 +1,199 @@ +/* Global box-sizing for predictable sizing */ +*, *::before, *::after { + box-sizing: border-box; +} + +h2 { + text-align: center; + /* Fluid font size between 20px and 24px */ + font-size: clamp(20px, 3vw, 24px); + color: #333; + margin-bottom: 20px; +} + +/* Form Styling */ +.info { + width: 60%; + max-width: 900px; /* optional max-width */ + background: #fff; + padding: 20px; + border-radius: 8px; + box-shadow: 0px 4px 10px rgba(0, 0, 0, 0.1); + display: flex; + flex-direction: column; + gap: 15px; + margin: 0 auto; /* center horizontally */ +} + +/* Responsive Form Layout */ +.row1, +.row2, +.row3 { + display: grid; + gap: 15px; +} + +.row2, +.row3 { + grid-template-columns: repeat(auto-fit, minmax(250px, 1fr)); +} + +label { + font-weight: bold; + display: block; + margin-bottom: 6px; +} + +input, +textarea, +select { + width: 100%; + padding: 10px; + border: 1px solid #ccc; + border-radius: 6px; + /* Fluid font size between 14px and 16px */ + font-size: clamp(14px, 1.5vw, 16px); +} + +/* Table styling */ +.table-wrapper { + overflow-x: auto; +} + +table { + width: 100%; + border-collapse: collapse; + margin: 20px 0; +} + +table th, +table td { + border: 1px solid #ddd; + padding: 8px; + text-align: center; + /* Fluid font size between 10px and 14px */ + font-size: clamp(10px, 1vw, 14px); +} + +table th { + background-color: #007bff; + color: white; + font-weight: bold; +} + +table tr:nth-child(even) { + background-color: #f2f2f2; +} + +table tr:hover { + background-color: #ddd; +} + +.download-btn { + display: inline-block; + padding: 10px 15px; + background-color: #28a745; + color: white; + text-decoration: none; + border-radius: 5px; + margin-top: 15px; + /* Fluid font size between 12px and 16px */ + font-size: clamp(12px, 1.5vw, 16px); +} + +.total-table { + width: 35%; +} + +/* Responsive Design */ +@media (max-width: 1024px) { + .info { + padding: 15px; + width: 80%; + } + + .row2, + .row3 { + grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); + } + + .invoice-table th, + .invoice-table td { + font-size: 12px; + padding: 8px; + } +} + +@media (max-width: 768px) { + .info { + width: 90%; + } + .row1, + .row2, + .row3 { + grid-template-columns: 1fr; + } +} + +@media (max-width: 600px) { + .table-wrapper { + overflow-x: auto; + } + + table { + min-width: 600px; + } +} + +@media (max-width: 480px) { + body { + max-width: 100%; + padding: 10px; + } + + .info { + padding: 10px; + width: 100%; + } + + table th, + table td { + font-size: 10px; + padding: 6px; + } +} + +@media (max-width: 360px) { + h2 { + font-size: 20px; + margin-bottom: 15px; + } + + .info { + width: 100%; + padding: 8px; + gap: 10px; + } + + input, + textarea, + select { + font-size: 14px; + padding: 8px; + } + + table th, + table td { + font-size: 8px; + padding: 4px; + } + + .total-table { + width: 100%; + } + + .download-btn { + padding: 8px 12px; + font-size: 14px; + } +} diff --git a/static/css/upload_excel_file.css b/static/css/upload_excel_file.css new file mode 100644 index 0000000..e9bc1b6 --- /dev/null +++ b/static/css/upload_excel_file.css @@ -0,0 +1,111 @@ +body { + font-family: Arial, sans-serif; + margin: 0; + padding: 0; + display: flex; + justify-content: center; + align-items: center; + min-height: 100vh; + background-color: #f9f9f9; +} + +.container { + max-width: 500px; + width: 90%; + background: #ffffff; + padding: 20px; + border-radius: 8px; + box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2); + text-align: center; +} + +h1 { + margin-bottom: 20px; + color: #333; +} + +form { + margin-bottom: 20px; +} + +input[type="file"] { + display: block; + margin: 0 auto 15px auto; + font-size: 18px; + padding: 10px; + cursor: pointer; + border: 2px solid #007bff; + border-radius: 5px; + background-color: #f4f4f4; +} + +input[type="file"]:hover { + border-color: #0056b3; +} + +button { + background-color: #4CAF50; + color: white; + border: none; + padding: 15px 25px; + font-size: 18px; + border-radius: 5px; + cursor: pointer; +} + +button:hover { + background-color: #45a049; +} + +/* Style for the Back Button */ +.back-button { + background-color: red; + color: white; + padding: 10px 20px; + text-decoration: none; + font-weight: bold; + border-radius: 5px; + display: inline-block; + margin-top: 20px; + text-align: center; +} + +.back-button:hover { + background-color: darkred; +} + +@media screen and (max-width: 600px) { + .container { + padding: 15px; + } + h1 { + font-size: 20px; + } + input[type="file"] { + font-size: 16px; + padding: 8px; + } + button, .back-button { + font-size: 14px; + padding: 10px 20px; + } +} +@media screen and (max-width: 360px) { + .container { + width: 95%; + padding: 10px; + } + h1 { + font-size: 18px; + margin-bottom: 15px; + } + input[type="file"] { + font-size: 14px; + padding: 6px; + } + button, .back-button { + font-size: 12px; + padding: 8px 16px; + } +} + diff --git a/static/download/Contractor_Report_1.xlsx b/static/download/Contractor_Report_1.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..82b0204de22981881a028931a535b7b3e98f51a5 GIT binary patch literal 6209 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 S0002=yZgyqIo~7xHTyq?e*KaF literal 0 HcmV?d00001 diff --git a/static/download/Contractor_Report_2.xlsx b/static/download/Contractor_Report_2.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..c38616c49d152c0a99c9f54c5fa7a171136d8163 GIT binary patch literal 8449 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 Pef$bNrt`|@zn=aFMMZTt literal 0 HcmV?d00001 diff --git a/static/download/Contractor_Report_3.xlsx b/static/download/Contractor_Report_3.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..323291658a5ddc6042a37d224c862031f8cdf5b0 GIT binary patch literal 39071 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 EAG`pW>;M1& literal 0 HcmV?d00001 diff --git a/static/download/Contractor_Report_4.xlsx b/static/download/Contractor_Report_4.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..aab8bce880da7d9f7d9381505c7c72f5b4cbf04b GIT binary patch literal 9147 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 E0Z2(XtpET3 literal 0 HcmV?d00001 diff --git a/static/images/icons/pen_blue_icon.png b/static/images/icons/pen_blue_icon.png new file mode 100644 index 0000000000000000000000000000000000000000..2f0bf5f1db66eb00ad7b79b39902a6261fa78869 GIT binary patch literal 3015 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 PsDthC@!!SWK|A+Select District'); + + data.forEach(function (district) { + + districtDropdown.append( + '' + ); + + }); + + districtDropdown.prop('disabled', false); + }, + + error: function () { + alert('Error fetching districts. Please try again.'); + } + }); + + } else { + $('#district_Id').prop('disabled', true); + } + + }); + +}); \ No newline at end of file diff --git a/static/js/district.js b/static/js/district.js new file mode 100644 index 0000000..d466a57 --- /dev/null +++ b/static/js/district.js @@ -0,0 +1,62 @@ + window.onload = function () { + document.getElementById('district_Name').focus(); + }; + + + $(document).ready(function () { + $("#district_Name").on("input", function () { + let districtName = $(this).val(); + // Remove numbers and special characters automatically + let cleanedName = districtName.replace(/[^A-Za-z ]/g, ""); + $(this).val(cleanedName); + }); + + $("#district_Name, #state_Id").on("input change", function () { + let districtName = $("#district_Name").val().trim(); + let stateId = $("#state_Id").val(); + + if (districtName === "" || stateId === "") { + $("#districtMessage").text("").css("color", ""); + $("#submitButton").prop("disabled", true); + return; + } + + $.ajax({ + url: "/check_district", + type: "POST", + contentType: "application/json", + data: JSON.stringify({ district_Name: districtName, state_Id: stateId }), + success: function (response) { + if (response.status === "available") { + $("#districtMessage").text(response.message).css("color", "green"); + $("#submitButton").prop("disabled", false); + } + }, + error: function (xhr) { + if (xhr.status === 409) { + $("#districtMessage").text("District already exists!").css("color", "red"); + $("#submitButton").prop("disabled", true); + } else if (xhr.status === 400) { + $("#districtMessage").text("Invalid district name! Only letters are allowed.").css("color", "red"); + $("#submitButton").prop("disabled", true); + } + } + }); + }); + + $("#districtForm").on("submit", function (event) { + event.preventDefault(); + $.ajax({ + url: "/add_district", + type: "POST", + data: $(this).serialize(), + success: function (response) { + alert(response.message); + location.reload(); + }, + error: function (xhr) { + alert(xhr.responseJSON.message); + } + }); + }); + }); \ No newline at end of file diff --git a/static/js/edit_hold_type.js b/static/js/edit_hold_type.js new file mode 100644 index 0000000..2e3175e --- /dev/null +++ b/static/js/edit_hold_type.js @@ -0,0 +1,18 @@ + $("#updateHoldTypeForm").on("submit", function(event) { + event.preventDefault(); + let holdTypeId = $("#hold_type_id").val(); + let newHoldType = $("#edit_hold_type").val().trim(); + let reg = /^[A-Za-z]/; + + if (!reg.test(newHoldType)) { + alert("Hold Type must start with a letter."); + return; + } + + $.post(`/update_hold_type/${holdTypeId}`, { hold_type: newHoldType }, function(response) { + alert(response.message); + window.location.href = "/"; + }).fail(function(xhr) { + alert(xhr.responseJSON.message); + }); + }); \ No newline at end of file diff --git a/static/js/holdAmount.js b/static/js/holdAmount.js new file mode 100644 index 0000000..76aff10 --- /dev/null +++ b/static/js/holdAmount.js @@ -0,0 +1,95 @@ +$(document).ready(function () { + // Create a module to manage hold amounts + window.holdAmountModule = { + holdCount: 0, + holdTypes: [], + + init: function() { + this.loadHoldTypes(); + this.setupEventListeners(); + }, + + loadHoldTypes: function() { + $.ajax({ + url: '/get_hold_types', + method: 'GET', + dataType: 'json', + success: (data) => { + this.holdTypes = data; + $("#add_hold_amount").prop('disabled', false); + }, + error: (err) => { + console.error('Failed to load hold types', err); + } + }); + }, + + setupEventListeners: function() { + $("#add_hold_amount").click(() => this.addHoldAmountField()); + $(document).on("click", ".remove-hold", (e) => this.removeHoldAmountField(e)); + $(document).on("change", ".hold-type-dropdown", () => this.refreshDropdowns()); + $(document).on("input", "input[name='hold_amount[]']", () => this.triggerHoldAmountChanged()); + }, + + addHoldAmountField: function() { + this.holdCount++; + $("#hold_amount_container").append(` +
    + + + +
    + `); + this.refreshDropdowns(); + this.triggerHoldAmountChanged(); + }, + + removeHoldAmountField: function(e) { + const id = $(e.target).attr("data-id"); + $(`#${id}`).remove(); + this.refreshDropdowns(); + this.triggerHoldAmountChanged(); + }, + + generateOptions: function(selectedForThisDropdown = '') { + const selectedValues = $("select[name='hold_type[]']").map(function() { + return $(this).val(); + }).get(); + + let options = ''; + this.holdTypes.forEach((type) => { + if (!selectedValues.includes(type.hold_type) || type.hold_type === selectedForThisDropdown) { + options += ``; + } + }); + return options; + }, + + refreshDropdowns: function() { + $("select[name='hold_type[]']").each(function() { + const currentVal = $(this).val(); + $(this).html(window.holdAmountModule.generateOptions(currentVal)); + $(this).val(currentVal); + }); + }, + + getTotalHoldAmount: function() { + let total = 0; + $("input[name='hold_amount[]']").each(function() { + total += parseFloat($(this).val()) || 0; + }); + return total; + }, + + triggerHoldAmountChanged: function() { + const event = new Event('holdAmountChanged'); + document.dispatchEvent(event); + } + }; + + // Initialize the module + window.holdAmountModule.init(); +}); \ No newline at end of file diff --git a/static/js/hold_types.js b/static/js/hold_types.js new file mode 100644 index 0000000..492a824 --- /dev/null +++ b/static/js/hold_types.js @@ -0,0 +1,39 @@ +$(document).ready(function () { + $("#hold_type").on("input", function () { + let holdType = $(this).val().replace(/^\s+/, ""); + $(this).val(holdType); + + let reg = /^.+$/; // all pattern allow + + if (!reg.test(holdType)) { + $("#holdTypeMessage").text("Hold Type must start with a letter.").css("color", "red"); + $("#addButton").prop("disabled", true); + return; + } else { + $("#holdTypeMessage").text("").css("color", ""); + $("#addButton").prop("disabled", false); + } + }); + + $("#holdTypeForm").on("submit", function (event) { + event.preventDefault(); + $.post("/add_hold_type", $(this).serialize(), function (response) { + alert(response.message); + location.reload(); + }).fail(function (xhr) { + alert(xhr.responseJSON.message); + }); + }); + + $(".delete-button").on("click", function () { + let id = $(this).data("id"); + if (confirm("Are you sure?")) { + $.post(`/delete_hold_type/${id}`, function (response) { + alert(response.message); + location.reload(); + }).fail(function (xhr) { + alert(xhr.responseJSON.message); + }); + } + }); +}); diff --git a/static/js/invoice.js b/static/js/invoice.js new file mode 100644 index 0000000..d3e0778 --- /dev/null +++ b/static/js/invoice.js @@ -0,0 +1,271 @@ +// $(document).ready(function () { +// // =============================== +// // FORM / TABLE TOGGLE +// // =============================== +// if ($('#addForm').length && $('#addTable').length) { +// $('#addForm').show(); +// $('#addTable').hide(); + +// $('#addButton').click(function () { +// $('#addForm').show(); +// $('#addTable').hide(); +// }); + +// $('#displayButton').click(function () { +// $('#addForm').hide(); +// $('#addTable').show(); +// }); +// } + +// // =============================== +// // Subcontractor autocomplete +// // =============================== +// $("#subcontractor").keyup(function () { +// let query = $(this).val(); +// if (query !== "") { +// $.ajax({ +// url: "/search_subcontractor", +// method: "POST", +// data: { query: query }, +// success: function (data) { +// $("#subcontractor_list").fadeIn().html(data); +// } +// }); +// } else { +// $("#subcontractor_list").fadeOut(); +// } +// }); + +// $(document).on("click", "li", function () { +// $("#subcontractor").val($(this).text()); +// $("#subcontractor_id").val($(this).attr("data-id")); +// $("#subcontractor_list").fadeOut(); +// }); + +// // Focus +// if (document.getElementById('subcontractor')) { +// document.getElementById('subcontractor').focus(); +// } + +// // =============================== +// // ADD INVOICE +// // =============================== +// if ($('#invoiceForm').length && window.location.href.includes('add_invoice')) { +// $("#invoiceForm").on("submit", function (e) { +// e.preventDefault(); +// let formData = $(this).serialize(); + +// $.ajax({ +// url: '/add_invoice', +// method: 'POST', +// data: formData, +// dataType: 'json', +// success: function (response) { +// if (response.status === "success") { +// alert(response.message || "Invoice added successfully!"); +// $('#invoiceForm')[0].reset(); // clear form +// $('#addForm').hide(); +// $('#addTable').show(); // switch to table +// location.reload(); // optional refresh +// } else { +// alert(response.message || "Error adding invoice."); +// } +// }, +// error: function (xhr) { +// alert(xhr.responseJSON?.message || "Submission failed. Please try again."); +// } +// }); +// }); +// } +// // Example AJAX update function +// function updateInvoice(invoiceId, formElement) { +// const formData = $(formElement).serialize(); + +// $.ajax({ +// url: '/update_invoice/' + invoiceId, +// method: 'POST', +// data: formData, +// dataType: 'json', +// success: function(response) { +// if(response.status === "success") { +// alert(response.message || "Invoice updated successfully!"); + +// // ✅ Hide Add Form, Show Table +// $('#addForm').hide(); +// $('#addTable').show(); + +// // Optional: reload table or refresh page +// location.reload(); +// } else { +// alert(response.message || "Update failed. Please try again."); +// } +// }, +// error: function(xhr) { +// alert(xhr.responseJSON?.message || "Error updating invoice."); +// } +// }); +// } + +// // =============================== +// // DELETE INVOICE +// // =============================== +// function deleteInvoice(invoiceId, element) { +// if (!confirm("Are you sure you want to delete this invoice?")) return; + +// $.ajax({ +// url: '/delete_invoice/' + invoiceId, +// method: 'GET', +// dataType: 'json', +// success: function (response) { +// alert(response.message || "Invoice deleted successfully!"); +// if (element) $(element).closest("tr").remove(); +// }, +// error: function (xhr) { +// alert(xhr.responseJSON?.message || "Error deleting invoice. Please try again."); +// } +// }); +// } + +$(document).ready(function () { + // =============================== + // FORM / TABLE TOGGLE + // =============================== + if ($('#addForm').length && $('#addTable').length) { + // Default: show form, hide table + $('#addForm').show(); + $('#addTable').hide(); + + // ✅ Check URL hash to show table instead + if (window.location.hash === "#addTable") { + $('#addForm').hide(); + $('#addTable').show(); + } + + $('#addButton').click(function () { + $('#addForm').show(); + $('#addTable').hide(); + }); + + $('#displayButton').click(function () { + $('#addForm').hide(); + $('#addTable').show(); + }); + } + + // =============================== + // Subcontractor autocomplete + // =============================== + $("#subcontractor").keyup(function () { + let query = $(this).val(); + if (query !== "") { + $.ajax({ + url: "/search_subcontractor", + method: "POST", + data: { query: query }, + success: function (data) { + $("#subcontractor_list").fadeIn().html(data); + } + }); + } else { + $("#subcontractor_list").fadeOut(); + } + }); + + $(document).on("click", "li", function () { + $("#subcontractor").val($(this).text()); + $("#subcontractor_id").val($(this).attr("data-id")); + $("#subcontractor_list").fadeOut(); + }); + + // Focus + if (document.getElementById('subcontractor')) { + document.getElementById('subcontractor').focus(); + } + + // =============================== + // ADD INVOICE + // =============================== + if ($('#invoiceForm').length && window.location.href.includes('add_invoice')) { + $("#invoiceForm").on("submit", function (e) { + e.preventDefault(); + let formData = $(this).serialize(); + + $.ajax({ + url: '/add_invoice', + method: 'POST', + data: formData, + dataType: 'json', + success: function (response) { + if (response.status === "success") { + alert(response.message || "Invoice added successfully!"); + $('#invoiceForm')[0].reset(); // clear form + $('#addForm').hide(); + $('#addTable').show(); // switch to table + location.reload(); // optional refresh + } else { + alert(response.message || "Error adding invoice."); + } + }, + error: function (xhr) { + let msg = xhr.responseJSON?.message || "Submission failed. Please try again."; + alert(msg); + } + }); + }); + } + + // =============================== + // UPDATE INVOICE + // =============================== + function updateInvoice(invoiceId, formElement) { + const formData = $(formElement).serialize(); + + $.ajax({ + url: '/update_invoice/' + invoiceId, + method: 'POST', + data: formData, + dataType: 'json', + success: function(response) { + if(response.status === "success") { + alert(response.message || "Invoice updated successfully!"); + + // Redirect to Add Invoice page's table part + window.location.href = "/add_invoice#addTable"; + } else { + alert(response.message || "Update failed. Please try again."); + } + }, + error: function(xhr) { + let msg = xhr.responseJSON?.message || "Error updating invoice."; + alert(msg); + } + }); + } + window.updateInvoice = updateInvoice; // make globally accessible + + // =============================== + // DELETE INVOICE + // =============================== + function deleteInvoice(invoiceId, element) { + if (!confirm("Are you sure you want to delete this invoice?")) return; + + $.ajax({ + url: '/delete_invoice/' + invoiceId, + method: 'GET', + dataType: 'json', + success: function (response) { + if (response.status === "success") { + alert(response.message || "Invoice deleted successfully!"); + if (element) $(element).closest("tr").remove(); + } else { + alert(response.message || "Error deleting invoice."); + } + }, + error: function (xhr) { + let msg = xhr.responseJSON?.message || "Error deleting invoice. Please try again."; + alert(msg); + } + }); + } + window.deleteInvoice = deleteInvoice; // make globally accessible +}); \ No newline at end of file diff --git a/static/js/save_data_success.js b/static/js/save_data_success.js new file mode 100644 index 0000000..d3a570e --- /dev/null +++ b/static/js/save_data_success.js @@ -0,0 +1,39 @@ +document.addEventListener('DOMContentLoaded', function () { + const form = document.getElementById('saveForm'); + + form.addEventListener('submit', function (e) { + e.preventDefault(); // Prevent normal form submission + + const formData = new FormData(form); + + fetch('/save_data', { + method: 'POST', + body: formData, + }).then(response => response.json()).then(data => { + if (data.success) { + Swal.fire({ + icon: 'success', + title: 'Success!', + text: data.success, + showConfirmButton: true, + confirmButtonText: 'OK' + }).then(() => { + const redirectUrl = "{{ url_for('upload_excel_file') }}"; // Redirect after success pop + }); + } else if (data.error) { + Swal.fire({ + icon: 'error', + title: 'Error!', + text: data.error, + }); + } + }).catch(error => { + Swal.fire({ + icon: 'error', + title: 'Error!', + text: 'An unexpected error occurred.', + }); + console.error('Error:', error); + }); + }); +}); \ No newline at end of file diff --git a/static/js/save_excel_file.js b/static/js/save_excel_file.js new file mode 100644 index 0000000..a670d69 --- /dev/null +++ b/static/js/save_excel_file.js @@ -0,0 +1,21 @@ +$("#saveForm").on("submit", function (event) { + event.preventDefault(); + $.ajax({ + url: "/save_data", + type: "POST", + data: $(this).serialize(), + success: function (response) { + if (response.success) { + alert("Success: " + response.success); // Show success alert + window.location.href = "/upload_excel_file"; // Redirect to the upload page + } + }, + error: function (xhr) { + if (xhr.responseJSON && xhr.responseJSON.error) { + alert("Error: " + xhr.responseJSON.error); + } else { + alert("An unexpected error occurred. Please try again."); + } + } + }); + }); \ No newline at end of file diff --git a/static/js/searchContractor.js b/static/js/searchContractor.js new file mode 100644 index 0000000..bb0251c --- /dev/null +++ b/static/js/searchContractor.js @@ -0,0 +1,43 @@ +$(document).ready(function () { + function fetchResults() { + let formData = $('#search-form').serialize(); + + $.ajax({ + type: 'POST', + url: '/search_contractor', + data: formData, + success: function (data) { + let tableBody = $('#result-table tbody'); + tableBody.empty(); + + if (data.length === 0) { + tableBody.append('No data found'); + } else { + data.forEach(function (row) { + tableBody.append(` + +
    ${row.Contractor_Name} + ${row.PMC_No} + ${row.State_Name} + ${row.District_Name} + ${row.Block_Name} + ${row.Village_Name} + + `); + }); + } + }, + error: function (xhr) { + alert(xhr.responseJSON.error); + } + }); + } + + $('#search-form input').on('keyup change', function () { + fetchResults(); + }); + }); + +window.onload = function () { + document.getElementById('subcontractor_name').focus(); + }; \ No newline at end of file diff --git a/static/js/search_on_table.js b/static/js/search_on_table.js new file mode 100644 index 0000000..4b0ab1e --- /dev/null +++ b/static/js/search_on_table.js @@ -0,0 +1,132 @@ +// Search on table using search inpute options +function searchTable() { + let input = document.getElementById("searchBar").value.toLowerCase(); + let tables = document.querySelectorAll("table"); + + tables.forEach(table => { + let rows = table.querySelectorAll("tr"); + + rows.forEach((row, index) => { + if (index === 0) return; // header skip + + let text = row.textContent.toLowerCase(); + + row.style.display = text.includes(input) ? "" : "none"; + }); + }); +} + + + +// Common Sorting Script for Tables +function sortTable(n, dir) { + var table, rows, switching, i, x, y, shouldSwitch; + table = document.getElementById("sortableTable"); // Ensure your table has this ID + switching = true; + + while (switching) { + switching = false; + rows = table.rows; + + for (i = 1; i < (rows.length - 1); i++) { + shouldSwitch = false; + x = rows[i].getElementsByTagName("TD")[n]; + y = rows[i + 1].getElementsByTagName("TD")[n]; + + if (dir == "asc") { + if (x.innerHTML.toLowerCase() > y.innerHTML.toLowerCase()) { + shouldSwitch = true; + break; + } + } else if (dir == "desc") { + if (x.innerHTML.toLowerCase() < y.innerHTML.toLowerCase()) { + shouldSwitch = true; + break; + } + } + } + + if (shouldSwitch) { + rows[i].parentNode.insertBefore(rows[i + 1], rows[i]); + switching = true; + } + } +} + +// Attach sorting functionality to all sortable tables +document.addEventListener("DOMContentLoaded", function () { + // Find all elements with the class "sortable-header" + var sortableHeaders = document.querySelectorAll(".sortable-header"); + + sortableHeaders.forEach(function (header) { + // Attach click event for ascending sort + if (header.querySelector(".sort-asc")) { + header.querySelector(".sort-asc").addEventListener("click", function () { + var columnIndex = Array.from(header.parentNode.children).indexOf(header); + sortTable(columnIndex, "asc"); + }); + } + + // Attach click event for descending sort + if (header.querySelector(".sort-desc")) { + header.querySelector(".sort-desc").addEventListener("click", function () { + var columnIndex = Array.from(header.parentNode.children).indexOf(header); + sortTable(columnIndex, "desc"); + }); + } + }); +}); + + +// ADD & Dispaly screen show +document.addEventListener("DOMContentLoaded", function () { + const addButton = document.getElementById("addButton"); + const displayButton = document.getElementById("displayButton"); + const addForm = document.getElementById("addForm"); + const addTable = document.getElementById("addTable"); + + // Show "Add State" form by default + addForm.style.display = "block"; + addButton.classList.add("active-button"); // Optional: Add styling for active button + + addButton.addEventListener("click", function () { + addForm.style.display = "block"; + addTable.style.display = "none"; + addButton.classList.add("active-button"); + displayButton.classList.remove("active-button"); + }); + + displayButton.addEventListener("click", function () { + addForm.style.display = "none"; + addTable.style.display = "block"; + displayButton.classList.add("active-button"); + addButton.classList.remove("active-button"); + }); +}); + +document.addEventListener("DOMContentLoaded", function () { + + let tables = document.querySelectorAll("table"); + + tables.forEach(table => { + let header = table.querySelector("tr:first-child"); + + if (header) { + header.style.position = "sticky"; + header.style.top = "0"; + header.style.background = "#fff"; + header.style.zIndex = "2"; + } + + if (!table.parentElement.classList.contains("table-wrapper")) { + let wrapper = document.createElement("div"); + wrapper.classList.add("table-wrapper"); + wrapper.style.maxHeight = "65vh" + wrapper.style.overflowY = "auto"; + + table.parentNode.insertBefore(wrapper, table); + wrapper.appendChild(table); + } + }); + +}); \ No newline at end of file diff --git a/static/js/showSuccessAlert.js b/static/js/showSuccessAlert.js new file mode 100644 index 0000000..7adcf0c --- /dev/null +++ b/static/js/showSuccessAlert.js @@ -0,0 +1,8 @@ +function showSuccessAlert(event) { + event.preventDefault(); // Prevent form submission + document.getElementById("successPopup").style.display = "block"; + setTimeout(function() { + document.getElementById("successPopup").style.display = "none"; + event.target.submit(); // Submit the form after showing the message + }, 2000); + } \ No newline at end of file diff --git a/static/js/sorting.js b/static/js/sorting.js new file mode 100644 index 0000000..933f8d9 --- /dev/null +++ b/static/js/sorting.js @@ -0,0 +1,66 @@ +$(document).ready(function () { + function fetchResults(sortBy = '', sortOrder = '') { + let formData = $('#search-form').serialize(); + formData += &sort_by=${sortBy}&sort_order=${sortOrder}; + + $.ajax({ + type: 'POST', + url: '/search_contractor', + data: formData, + success: function (data) { + let tableBody = $('#result-table tbody'); + tableBody.empty(); + + if (data.length === 0) { + tableBody.append('No data found'); + } else { + data.forEach(function (row) { + tableBody.append(` + + ${row.Contractor_Name} + ${row.PMC_No} + ${row.State_Name} + ${row.District_Name} + ${row.Block_Name} + ${row.Village_Name} + + `); + }); + } + }, + error: function (xhr) { + alert(xhr.responseJSON.error); + } + }); + } + + $('#search-form input').on('keyup change', function () { + fetchResults(); + }); + + function showSortOptions(thElement, column) { + let sortMenu = $('#sort-options'); + let offset = $(thElement).position(); + let thHeight = $(thElement).outerHeight(); + + sortMenu.html(` + + + `); + + sortMenu.css({ + display: 'block', + top: offset.top + thHeight + 'px', + left: offset.left + 'px' + }); + } + + $(document).click(function(event) { + if (!$(event.target).closest('.sort-options, th').length) { + $('#sort-options').hide(); + } + }); + + window.fetchResults = fetchResults; + window.showSortOptions = showSortOptions; + }); \ No newline at end of file diff --git a/static/js/state.js b/static/js/state.js new file mode 100644 index 0000000..21625f8 --- /dev/null +++ b/static/js/state.js @@ -0,0 +1,61 @@ + window.onload = function () { + document.getElementById('state_Name').focus(); + }; + + +$(document).ready(function () { + $("#state_Name").on("input", function () { + let stateName = $(this).val(); + // Remove numbers and special characters automatically + let cleanedName = stateName.replace(/[^A-Za-z ]/g, ""); + $(this).val(cleanedName); + }); + + $("#state_Name").on("input", function () { + let stateName = $("#state_Name").val().trim(); + + if (stateName === "") { + $("#stateMessage").text("").css("color", ""); + $("#submitButton").prop("disabled", true); + return; + } + + $.ajax({ + url: "/check_state", + type: "POST", + contentType: "application/json", + data: JSON.stringify({ state_Name: stateName }), + success: function (response) { + if (response.status === "available") { + $("#stateMessage").text(response.message).css("color", "green"); + $("#submitButton").prop("disabled", false); + } + }, + error: function (xhr) { + if (xhr.status === 409) { + $("#stateMessage").text("State already exists!").css("color", "red"); + $("#submitButton").prop("disabled", true); + } else if (xhr.status === 400) { + $("#stateMessage").text("Invalid state name! Only letters are allowed.").css("color", "red"); + $("#submitButton").prop("disabled", true); + } + } + }); + }); + + $("#stateForm").on("submit", function (event) { + event.preventDefault(); + $.ajax({ + url: "/add_state", + type: "POST", + data: $(this).serialize(), + success: function (response) { + alert(response.message); + location.reload(); + }, + error: function (xhr) { + alert(xhr.responseJSON.message); + } + }); + }); +}); \ No newline at end of file diff --git a/static/js/subcontractor.js b/static/js/subcontractor.js new file mode 100644 index 0000000..33cdb43 --- /dev/null +++ b/static/js/subcontractor.js @@ -0,0 +1,49 @@ +function validateInput() { + let isValid = true; + + // Get form elements + let contractorName = document.getElementById("Contractor_Name").value; + let mobileNo = document.getElementById("Mobile_No").value; + let panNo = document.getElementById("PAN_No").value; + let email = document.getElementById("Email").value; + let passwordField = document.getElementById("Contractor_password"); + let submitBtn = document.getElementById("submitBtn"); + + // Validation patterns + let mobileRegex = /^[0-9]{10}$/; + let panRegex = /^[A-Z0-9]{10}$/; + let emailRegex = /^[a-z]+@[a-z]+\.[a-z]{2,6}$/; + + // Validate Mobile No + if (!mobileNo.match(mobileRegex)) { + document.getElementById("mobileError").innerText = "Mobile No must be exactly 10 digits."; + isValid = false; + } else { + document.getElementById("mobileError").innerText = ""; + } + + // Validate PAN No + if (!panNo.match(panRegex)) { + document.getElementById("panError").innerText = "PAN No must be uppercase letters or digits (10 chars)."; + isValid = false; + } else { + document.getElementById("panError").innerText = ""; + } + + // Validate Email + if (!email.match(emailRegex)) { + document.getElementById("emailError").innerText = "Email must be lowercase, contain '@' and '.'"; + isValid = false; + } else { + document.getElementById("emailError").innerText = ""; + } + + + + // Enable or disable the submit button + submitBtn.disabled = !isValid; + } + +window.onload = function () { + document.getElementById('Contractor_Name').focus(); + }; \ No newline at end of file diff --git a/static/js/validateFileInput.js b/static/js/validateFileInput.js new file mode 100644 index 0000000..ab6b9ad --- /dev/null +++ b/static/js/validateFileInput.js @@ -0,0 +1,12 @@ +function validateFileInput() { + const fileInput = document.querySelector('input[type="file"]'); + const filePath = fileInput.value; + const allowedExtensions = /(\.xlsx|\.xls)$/i; + + if (!allowedExtensions.exec(filePath)) { + alert("Please upload a valid Excel file (.xlsx or .xls only)."); + fileInput.value = ''; + return false; + } + return true; + } \ No newline at end of file diff --git a/static/js/village.js b/static/js/village.js new file mode 100644 index 0000000..2a6f4fd --- /dev/null +++ b/static/js/village.js @@ -0,0 +1,250 @@ +window.onload = function () { + if (document.getElementById('Village_Name')) { + document.getElementById('Village_Name').focus(); + } +}; + +$(document).ready(function () { + + // ✅ RUN ONLY IF THIS PAGE HAS FORM/TABLE + if ($('#addForm').length && $('#addTable').length) { + + // ✅ DEFAULT VIEW → SHOW FORM + $('#addForm').show(); + $('#addTable').hide(); + + + // 🔥 BUTTON TOGGLE + $('#addButton').click(function () { + $('#addForm').show(); + $('#addTable').hide(); + }); + + $('#displayButton').click(function () { + $('#addForm').hide(); + $('#addTable').show(); + }); + } + + + // =============================== + // STATE → DISTRICT + // =============================== + if ($('#state_Id').length) { + + $('#state_Id').change(function () { + + var stateId = $(this).val(); + + if (stateId) { + + $.ajax({ + url: '/get_districts/' + stateId, + type: 'GET', + + success: function (data) { + + var districtDropdown = $('#district_Id'); + + districtDropdown.empty(); + districtDropdown.append(''); + + data.forEach(function (district) { + districtDropdown.append( + '' + ); + }); + + districtDropdown.prop('disabled', false); + } + }); + } + }); + } + + + // =============================== + // DISTRICT → BLOCK + // =============================== + if ($('#district_Id').length) { + + $('#district_Id').change(function () { + + var districtId = $(this).val(); + + if (districtId) { + + $.ajax({ + url: '/get_blocks/' + districtId, + type: 'GET', + + success: function (data) { + + var blockDropdown = $('#block_Id'); + + blockDropdown.empty(); + blockDropdown.append(''); + + data.forEach(function (block) { + blockDropdown.append( + '' + ); + }); + + blockDropdown.prop('disabled', false); + } + }); + } + }); + } + + + // =============================== + // VILLAGE NAME VALIDATION + // =============================== + if ($('#Village_Name').length) { + + $('#Village_Name').on('input', function () { + + var villageName = $(this).val(); + var validPattern = /^[A-Za-z ]*$/; + + if (!validPattern.test(villageName)) { + + $('#villageMessage') + .text('Only letters and spaces are allowed!') + .css('color', 'red'); + + $('#submitVillage').prop('disabled', true); + + } else { + + $('#villageMessage').text(''); + $('#submitVillage').prop('disabled', false); + } + }); + } + + + // =============================== + // CHECK DUPLICATE VILLAGE + // =============================== + if ($('#Village_Name').length && $('#block_Id').length) { + + $('#Village_Name, #block_Id').on('change keyup', function () { + + var blockId = $('#block_Id').val(); + var villageName = $('#Village_Name').val().trim(); + + if (blockId && villageName) { + + $.ajax({ + url: '/check_village', + type: 'POST', + + data: { + block_Id: blockId, + Village_Name: villageName + }, + + success: function (response) { + + if (response.status === 'exists') { + + $('#villageMessage') + .text(response.message) + .css('color', 'red'); + + $('#submitVillage').prop('disabled', true); + + } else { + + $('#villageMessage') + .text(response.message) + .css('color', 'green'); + + $('#submitVillage').prop('disabled', false); + } + }, + + error: function () { + + $('#villageMessage') + .text('Error checking village name') + .css('color', 'red'); + + $('#submitVillage').prop('disabled', true); + } + }); + } + }); + } + + + // =============================== + // ADD VILLAGE (SAFE SCOPE FIX) + // =============================== + if ($('#villageForm').length) { + + $('#villageForm').submit(function (event) { + + event.preventDefault(); + + $.ajax({ + url: '/add_village', + type: 'POST', + data: $(this).serialize(), + + success: function (response) { + + if (response && response.status === 'success') { + + alert(response.message || 'Village added successfully!'); + + // ✅ clear form + $('#villageForm')[0].reset(); + + // ✅ switch to table + $('#addForm').hide(); + $('#addTable').show(); + + // optional refresh + location.reload(); + + } else { + alert(response.message || 'Error adding village. Please try again.'); + } + }, + + error: function () { + alert('An error occurred. Please try again.'); + } + }); + + }); + } + +}); + + + + +// =============================== +// DELETE FUNCTION (SAFE) +// =============================== +function deleteVillage(villageId, element) { + if (!confirm("Are you sure you want to delete this village?")) return; + + $.ajax({ + url: '/delete_village/' + villageId, + type: 'GET', + dataType: 'json', + success: function (response) { + alert(response.message); // ✅ now shows "Village deleted successfully!" + if (element) $(element).closest("tr").remove(); + }, + error: function () { + alert("Error deleting village. Please try again."); + } + }); +} \ No newline at end of file diff --git a/templates/activity_log.html b/templates/activity_log.html new file mode 100644 index 0000000..e3d6ade --- /dev/null +++ b/templates/activity_log.html @@ -0,0 +1,116 @@ + + + + + + Activity Logs + + + + +

    Activity Logs

    + +
    + + + + + + + + + + + +
    + + + + + + + + + {% for log in logs %} + + + + + + + {% endfor %} + {% if logs|length == 0 %} + + + + {% endif %} +
    TimestampUserActionDetails
    {{ log.timestamp }}{{ log.user }}{{ log.action }}{{ log.details }}
    No logs found
    + + + + + \ No newline at end of file diff --git a/templates/add_block.html b/templates/add_block.html new file mode 100644 index 0000000..4bcf1f7 --- /dev/null +++ b/templates/add_block.html @@ -0,0 +1,99 @@ +{% extends 'base.html' %} +{% block content %} + + + + + Add Block + + + + + + + + +
    + + +
    + + + + + +{% endblock %} \ No newline at end of file diff --git a/templates/add_district.html b/templates/add_district.html new file mode 100644 index 0000000..ceeaee6 --- /dev/null +++ b/templates/add_district.html @@ -0,0 +1,85 @@ +{% extends 'base.html' %} +{% block content %} + + Add District + + + + + + + +
    + + +
    + + + + + + +{% endblock %} \ No newline at end of file diff --git a/templates/add_gst_release.html b/templates/add_gst_release.html new file mode 100644 index 0000000..a037dca --- /dev/null +++ b/templates/add_gst_release.html @@ -0,0 +1,209 @@ +{% extends 'base.html' %} +{% block content %} + + + + Add GST Release + + + + + + + +
    + + +
    + + + + +
    + {% with messages = get_flashed_messages(with_categories=true) %} + {% if messages %} + {% for category, message in messages %} +
    + {{ message }} +
    + {% endfor %} + {% endif %} + {% endwith %} +
    + + + + + + + + +{% endblock %} \ No newline at end of file diff --git a/templates/add_hold_type.html b/templates/add_hold_type.html new file mode 100644 index 0000000..a65d0e4 --- /dev/null +++ b/templates/add_hold_type.html @@ -0,0 +1,73 @@ +{% extends 'base.html' %} +{% block content %} + + + Manage Hold Types + + + + + + + + +
    + + +
    + +
    +

    Add Hold Types

    +
    + + + + +
    +
    + + + +{% endblock %} \ No newline at end of file diff --git a/templates/add_invoice.html b/templates/add_invoice.html new file mode 100644 index 0000000..e6372e9 --- /dev/null +++ b/templates/add_invoice.html @@ -0,0 +1,536 @@ +{% extends 'base.html' %} {% block content %} + + + + Add Invoice + + + + + + + {% if success == 'true' %} + + {% endif %} + + + {% with messages = get_flashed_messages(with_categories=true) %} {% if + messages %} +
    + {% for category, message in messages %} +
    {{ message }}
    + {% endfor %} +
    + {% endif %} {% endwith %} + +
    + + +
    + + + + + + +{% endblock %} diff --git a/templates/add_payment.html b/templates/add_payment.html new file mode 100644 index 0000000..004796b --- /dev/null +++ b/templates/add_payment.html @@ -0,0 +1,207 @@ +{% extends 'base.html' %} +{% block content %} + + + + + Add Payment + + + + + + + +
    + + +
    + + + +
    + Payment added successfully! +
    + + + + + + + + + + +{% endblock %} \ No newline at end of file diff --git a/templates/add_purchase_order.html b/templates/add_purchase_order.html new file mode 100644 index 0000000..4e57849 --- /dev/null +++ b/templates/add_purchase_order.html @@ -0,0 +1,147 @@ +{% extends 'base.html' %} +{% block content %} + + Manage Purchases + + + + +
    + + +
    + + + + + + + + + + + + +{% endblock %} diff --git a/templates/add_state.html b/templates/add_state.html new file mode 100644 index 0000000..705b42e --- /dev/null +++ b/templates/add_state.html @@ -0,0 +1,74 @@ +{% extends 'base.html' %} +{% block content %} + + + Add State + + + + + + + + +
    + + +
    + + + + + + + +{% endblock %} \ No newline at end of file diff --git a/templates/add_subcontractor.html b/templates/add_subcontractor.html new file mode 100644 index 0000000..7bef52e --- /dev/null +++ b/templates/add_subcontractor.html @@ -0,0 +1,127 @@ +{% extends 'base.html' %} +{% block content %} + + + SubContractor + + + + + + + + +
    + + +
    + + + + + +{% endblock %} diff --git a/templates/add_village.html b/templates/add_village.html new file mode 100644 index 0000000..0c4c4d5 --- /dev/null +++ b/templates/add_village.html @@ -0,0 +1,129 @@ +{% extends 'base.html' %} {% block content %} + + + + Village Management + + + + + + +
    + + +
    + + + + + +{% with messages = get_flashed_messages(with_categories=true) %} + {% if messages %} + + {% endif %} +{% endwith %} + +{% endblock %} diff --git a/templates/add_work_order.html b/templates/add_work_order.html new file mode 100644 index 0000000..ad5ad38 --- /dev/null +++ b/templates/add_work_order.html @@ -0,0 +1,137 @@ +{% extends 'base.html' %} +{% block content %} + + Manage Work Order + + + + + + + +
    + + +
    +
    + +
    +{# display data #} + + + + + + + +{% endblock %} \ No newline at end of file diff --git a/templates/admin_profile.html b/templates/admin_profile.html new file mode 100644 index 0000000..fbd60ca --- /dev/null +++ b/templates/admin_profile.html @@ -0,0 +1,49 @@ +{% extends 'base.html' %} +{% block content %} + + + + Admin Profile + + +
    +

    Admin Profile

    + {% with messages = get_flashed_messages(with_categories=true) %} + {% if messages %} + {% for category, message in messages %} +

    {{ message }}

    + {% endfor %} + {% endif %} + {% endwith %} +
    +
    + + +
    + + + + + + + + + + + +
    +
    + + +{% endblock %} diff --git a/templates/base.html b/templates/base.html new file mode 100644 index 0000000..bb52864 --- /dev/null +++ b/templates/base.html @@ -0,0 +1,97 @@ + + + + + + {% block title %}Payment Reconciliation{% endblock %} + + + + + + + + + + + + + +
    + {% block content %}{% endblock %} +
    + + + + + \ No newline at end of file diff --git a/templates/edit_block.html b/templates/edit_block.html new file mode 100644 index 0000000..86a5059 --- /dev/null +++ b/templates/edit_block.html @@ -0,0 +1,60 @@ + + + +{% extends 'base.html' %} +{% block content %} + + Edit Block + + + + + +

    Edit Block

    + + +{% with messages = get_flashed_messages(with_categories=true) %} + {% if messages %} + {% for category, message in messages %} + + {% endfor %} + {% endif %} +{% endwith %} + + +
    + +

    + + +

    + + +

    + + +
    + + +{% endblock %} diff --git a/templates/edit_district.html b/templates/edit_district.html new file mode 100644 index 0000000..c028514 --- /dev/null +++ b/templates/edit_district.html @@ -0,0 +1,26 @@ +{% extends 'base.html' %} +{% block content %} + + Edit District + + + +

    Edit District

    + +
    + + + + + +
    + + + + +{% endblock %} \ No newline at end of file diff --git a/templates/edit_grn.html b/templates/edit_grn.html new file mode 100644 index 0000000..22f0200 --- /dev/null +++ b/templates/edit_grn.html @@ -0,0 +1,39 @@ + + +Edit GRN + + + +

    Edit GRN

    +
    + +

    + + +

    + + +

    + + +

    + + +

    + + +

    + + +

    + + +

    + + +

    + + +
    + + diff --git a/templates/edit_gst_release.html b/templates/edit_gst_release.html new file mode 100644 index 0000000..1392923 --- /dev/null +++ b/templates/edit_gst_release.html @@ -0,0 +1,45 @@ +{% extends 'base.html' %} +{% block content %} + + + + Edit GST Release + + + +

    Edit GST Release

    + +
    + + +
    +

    + + +
    +

    + + +
    +

    + + +
    +

    + + +
    +

    + + +
    +

    + + + + + +
    + + +{% endblock %} \ No newline at end of file diff --git a/templates/edit_hold_type.html b/templates/edit_hold_type.html new file mode 100644 index 0000000..a0a2123 --- /dev/null +++ b/templates/edit_hold_type.html @@ -0,0 +1,57 @@ +{% extends 'base.html' %} + +{% block content %} + + + Edit Hold Type + + + + + +

    Edit Hold Type

    +
    + + + + + +
    + + + +{% endblock %} \ No newline at end of file diff --git a/templates/edit_invoice.html b/templates/edit_invoice.html new file mode 100644 index 0000000..f8ab8f8 --- /dev/null +++ b/templates/edit_invoice.html @@ -0,0 +1,226 @@ +{% extends 'base.html' %} +{% block content %} + + + + + Edit Invoice + + + + + + + + +
    +

    Edit Invoice

    + + + + +
    + + +
    +
    + + + + +
    +
    + + +
    +
    + + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    + + +
    + {% set seen_types = [] %} + {% for hold in invoice.hold_amounts %} + {% if hold.hold_type not in seen_types %} + {% set _ = seen_types.append(hold.hold_type) %} +
    + + + + + +
    + {% endif %} + {% endfor %} +
    + + +
    + +
    + + +
    +
    + + +
    +
    + + +
    +
    + + + +
    +
    + + + + + +{% endblock %} \ No newline at end of file diff --git a/templates/edit_payment.html b/templates/edit_payment.html new file mode 100644 index 0000000..00f7d89 --- /dev/null +++ b/templates/edit_payment.html @@ -0,0 +1,49 @@ +{% extends 'base.html' %} +{% block content %} + + + + Edit Payment + + + + +

    Edit Payment

    + +
    + + + + + + + + + +
    +

    + +
    +

    + + +
    +

    + +
    +

    + +
    +

    + +
    +

    + + +
    + + +{% endblock %} diff --git a/templates/edit_purchase.html b/templates/edit_purchase.html new file mode 100644 index 0000000..bfa090a --- /dev/null +++ b/templates/edit_purchase.html @@ -0,0 +1,47 @@ + + + + Edit Purchase + + + +

    Edit Purchase Order

    +
    + +

    + + +

    + + +

    + + +

    + + +

    + + +

    + + +

    + + +

    + + + +

    + + +

    + + +

    + + +
    + + diff --git a/templates/edit_state.html b/templates/edit_state.html new file mode 100644 index 0000000..144fdfe --- /dev/null +++ b/templates/edit_state.html @@ -0,0 +1,17 @@ +{% extends 'base.html' %} +{% block content %} + + Edit State + + + +

    Edit State

    + +
    + + +
    + + + +{% endblock %} \ No newline at end of file diff --git a/templates/edit_subcontractor.html b/templates/edit_subcontractor.html new file mode 100644 index 0000000..3238527 --- /dev/null +++ b/templates/edit_subcontractor.html @@ -0,0 +1,50 @@ +{% extends 'base.html' %} +{% block content %} + + + Edit SubContractor + + + + +

    Edit Sub-Contractor

    +
    + + + +
    + + +
    + + +
    + + +
    + + +
    + + +
    + + +
    + + +
    + + +
    + + +
    + + + +{% endblock %} \ No newline at end of file diff --git a/templates/edit_village.html b/templates/edit_village.html new file mode 100644 index 0000000..f2b4dc2 --- /dev/null +++ b/templates/edit_village.html @@ -0,0 +1,44 @@ +{% extends 'base.html' %} +{% block content %} + + + + Edit Village + + + + +
    +
    +

    Edit Village

    +
    + + + + + + + + +
    + + + + + +{% endblock %} diff --git a/templates/grn_form.html b/templates/grn_form.html new file mode 100644 index 0000000..eae3847 --- /dev/null +++ b/templates/grn_form.html @@ -0,0 +1,77 @@ + + + + GRN Management + + + +

    Add GRN

    +
    + +

    + + +

    +{# #} +{# #} + + +

    + + +

    + + +

    + + +

    + + +

    + + +

    + + +

    + + +
    + +

    All GRNs

    + + + + + + {% for grn in grns %} + + + + + + + + + + + + + + {% endfor %} +
    IDDatePurchase IDSupplierItemQtyUnitRateAmountRemarksActions
    {{ grn[0] }}{{ grn[1] }}{{ grn[2] }}{{ grn[3] }}{{ grn[4] }}{{ grn[5] }}{{ grn[6] }}{{ grn[7] }}{{ grn[8] }}{{ grn[9] }} + Edit + +
    + +
    +
    + + diff --git a/templates/index.html b/templates/index.html new file mode 100644 index 0000000..bc8094b --- /dev/null +++ b/templates/index.html @@ -0,0 +1,164 @@ + + + + + + + Payment Reconciliation + + + + + + +
    + Logout +
    + +
    + +

    Laxmi Civil Engineering Services Pvt. Ltd.

    +
    +

    Payment Reconciliation

    +
    +
    + + + + + \ No newline at end of file diff --git a/templates/login.html b/templates/login.html new file mode 100644 index 0000000..e638b55 --- /dev/null +++ b/templates/login.html @@ -0,0 +1,90 @@ + + + + + LCEPL Payment Reconciliation + + + + + + diff --git a/templates/pmc_report.html b/templates/pmc_report.html new file mode 100644 index 0000000..c218c5f --- /dev/null +++ b/templates/pmc_report.html @@ -0,0 +1,340 @@ + + +{% extends 'base.html' %} +{% block content %} + + + + PMC Report + + + + + +
    +

    PMC Report

    + +
    +

    Contractor Details

    +
    +
    + + +
    +
    + + +
    +
    +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    +
    +
    + + + +
    +
    + + +
    +
    +

    PMC Report for PMC No: {{ info.PMC_No}}

    +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    +
    +
    + + +
    +
    + + +
    +
    +
    +
    + + +

    Invoice Details

    + + + + + + + + + + + + + + + + + + + + + {% set hold_types = invoices | map(attribute='hold_type') | reject('none') | unique | list %} + {% for hold in hold_types %} + + {% endfor %} + + + + + + + {% if invoices %} + {% for invoice in invoices %} + + + + + + + + + + + + + + + + + + + {% for hold in hold_types %} + + {% endfor %} + + + + + + {% endfor %} + + + + + + + + + + + + {% set hold_types = invoices | map(attribute='hold_type') | unique | list %} + {% for hold in hold_types %} + + {% endfor %} + + + + + {% else %} + + + + {% endif %} + +
    PMC NoVillage NameWork TypeInvoice DetailsInvoice DateInvoice NoBasic AmountDebitAfter Debit AmtGST (18%)AmountTDS (1%)SD (5%)On CommissionHydro Testing{{ hold }}GST SD (18%)Final Amount
    {{ invoice.PMC_No }}{{ invoice.Village_Name.capitalize() }}{{ invoice.Work_Type }}{{ invoice.Invoice_Details }}{{ invoice.Invoice_Date }}{{ invoice.Invoice_No }}{{ invoice.Basic_Amount }}{{ invoice.Debit_Amount }}{{ invoice.After_Debit_Amount }}{{ invoice.GST_Amount }}{{ invoice.Amount }}{{ invoice.TDS_Amount }}{{ invoice.SD_Amount }}{{ invoice.On_Commission }}{{ invoice.Hydro_Testing }} + {% if invoice.hold_type == hold %} + {{ invoice.hold_amount }} + {% else %} + 0 + {% endif %} + {{ invoice.GST_SD_Amount }}{{ invoice.Final_Amount }}
    Total{{total["sum_invo_basic_amt"]}}{{total["sum_invo_debit_amt"]}}{{total["sum_invo_after_debit_amt"]}}{{total["sum_invo_gst_amt"]}}{{total["sum_invo_amt"]}}{{total["sum_invo_tds_amt"]}}{{total["sum_invo_ds_amt"]}}{{total["sum_invo_on_commission"]}}{{total["sum_invo_hydro_test"]}}{{total["sum_invo_hold_amt"]}}{{total["sum_invo_gst_sd_amt"]}}{{total["sum_invo_final_amt"]}}
    No invoices found.
    +

    Hold Release

    + + + + + + + + + + + + + {%if hold_release%} + {%for hold in hold_release%} + + + + + + + + + {%endfor%} + {%else%} + + + + {%endif%} + +
    PMC NoInvoice NoInvoice DetailsBasic AmountTotal AmountUTR
    {{ hold['PMC_No'] }}{{ hold['Invoice_No'] }}{{ hold['Invoice_Details'] }}{{ hold['Basic_Amount'] }}{{ hold['Total_Amount'] }}{{ hold['UTR'] }}
    No data present
    + +
    +

    GST Release Note Details

    + + + + + + + + + + + {% if gst_rel %} + {% for gst in gst_rel %} + + + + + + + {% endfor %} + + + + + + + {% else %} + + + + {% endif %} + +
    PMC NoInvoice NoBasic AmountFinal Amount
    {{ gst.pmc_no }}{{ gst.invoice_no }}{{ gst.basic_amount }}{{ gst.final_amount }}
    Total{{total["sum_gst_basic_amt"]}}{{total["sum_gst_final_amt"]}}
    No GST release found.
    + + +

    Credit Details

    + + + + + + + + + + + + + + + + + + + {% if credit_note %} + {% for credit in credit_note %} + + + + + + + + + + + + + + + + {% endfor %} + + {% else %} + + + + {% endif %} + + + +
    PMC NoInvoice DetailsBasic AmountDebitAfter Debit AmtGST AmountAmountFinal AmountPayment AmountTotal AmountUTR
    {{ credit["PMC_No"] }}{{ credit["Invoice_Details"] }}{{ credit["Basic_Amount"] }}{{ credit["Debit_Amount"] }}{{ credit["After_Debit_Amount"] }}{{ credit["GST_Amount"] }}{{ credit["Amount"] }}{{ credit["Final_Amount"] }}{{ credit["Payment_Amount"] }}{{ credit["Total_Amount"] }}{{ credit["UTR"] }}
    No Credit note found.
    + +
    +

    Payment Details

    + + + + + + + + + + + + + {% if payments %} + {% for pay in payments %} + + + + + + + + + {% endfor %} + + + + + + + + {% else %} + + + + + {% endif %} + + +
    PMC NoInvoice NoAmountTDS Amount @ 1% on BASIC AMOUNTTotal Amount PaidUTR
    {{ pay.pmc_no }}{{ pay.invoice_no }}{{ pay.Payment_Amount }}{{ pay.TDS_Payment_Amount }}{{ pay.Total_amount }}{{ pay.utr}}
    Total{{total["sum_pay_payment_amt"]}}{{total["sum_pay_tds_payment_amt"]}}{{total["sum_pay_total_amt"]}}
    No payment found.
    + + + + + +{% endblock %} \ No newline at end of file diff --git a/templates/purchase_order.html b/templates/purchase_order.html new file mode 100644 index 0000000..33b3c30 --- /dev/null +++ b/templates/purchase_order.html @@ -0,0 +1,10 @@ + + + + + Purchase Order + + + + + \ No newline at end of file diff --git a/templates/purchase_order_report.html b/templates/purchase_order_report.html new file mode 100644 index 0000000..aacea0c --- /dev/null +++ b/templates/purchase_order_report.html @@ -0,0 +1,205 @@ + + + + Purchase Order Report + + + + + +

    Purchase Order Report

    + +
    +
    + + +
    + +
    + + +
    + + + +
    + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/templates/report.html b/templates/report.html new file mode 100644 index 0000000..de53ef8 --- /dev/null +++ b/templates/report.html @@ -0,0 +1,108 @@ +{% extends 'base.html' %} +{% block content %} + + + + Contractor Search + + + + + +
    +

    Search Contractor Report

    +
    +
    +
    +
    + + +
    +
    + + +
    +
    +
    +
    + + +
    +
    + + +
    +
    +
    +
    + + +
    +
    + + +
    +
    +
    +
    + + +
    +
    + + +
    +
    +
    + +

    Contractor List

    + + + + + + + + + + + + + + +
    Contractor Name + + PMC NoState + + District + + Block + + Village + +
    +
    +
    + + +{% endblock %} \ No newline at end of file diff --git a/templates/show_excel_file.html b/templates/show_excel_file.html new file mode 100644 index 0000000..19b8903 --- /dev/null +++ b/templates/show_excel_file.html @@ -0,0 +1,85 @@ + + + + Excel Data + + + + + + + + +

    Excel Data

    + +

    File Information:

    +
      +
    • Subcontractor: {{ file_info['Subcontractor'] }}
    • +
    • State: {{ file_info['State'] }}
    • +
    • District: {{ file_info['District'] }}
    • +
    • Block: {{ file_info['Block'] }}
    • +
    + +{% if errors %} +
    +

    Validation Errors:

    +
      + {% for error in errors %} +
    • {{ error }}
    • + {% endfor %} +
    +
    +{% endif %} + +

    Data Table:

    +
    +
    + + + + + {% for var in variables %} + + {% endfor %} + + + + {% for row in data %} + + + {% for var in variables %} + + {% endfor %} + + {% endfor %} + +
    Row No{{ var }}
    {{ row['Row Number'] }}
    +
    + + + + + + + + + + + + + + + + + + + + +
    + +Back to Dashboard + + + + + \ No newline at end of file diff --git a/templates/subcontractor_report.html b/templates/subcontractor_report.html new file mode 100644 index 0000000..e1f5001 --- /dev/null +++ b/templates/subcontractor_report.html @@ -0,0 +1,372 @@ + + + +{% extends 'base.html' %} +{% block content %} + + + + Contractor Report + + + + +
    +

    Contractor Report

    + +
    +

    Contractor Details

    +
    +
    + + +
    +
    + + +
    +
    +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    +
    +
    + + + +
    +
    + + +
    +
    +
    +
    + + +
    +
    + + +
    +
    + + +
    +
    +
    + +

    Total

    + + + + + + + + + + + + + {% if hold_types %} + + {% for hold in hold_types %} + + + {% endfor %} + + {% endif %} + + + + +
    {{current_date}}
    Advance / Suplus{{total["sum_invo_final_amt"]+total["sum_gst_final_amt"]-total["sum_pay_total_amt"]}}
    {{ hold.hold_type }}{{total["sum_invo_hold_amt"]}}
    Amount With TDS{{total["sum_invo_tds_amt"]}}
    + + + + + + + + + + + +

    Invoice Details

    + + + + + + + + + + + + + + + + + + + + + {% set hold_types = invoices | map(attribute='hold_type') | unique | list %} + {% for hold in hold_types %} + + {% endfor %} + + + + + + + {% if invoices %} + {% for invoice in invoices %} + + + + + + + + + + + + + + + + + + + {% set hold_types = invoices | map(attribute='hold_type') | unique | list %} + {% for hold in hold_types %} + + {% endfor %} + + + + + + {% endfor %} + + + + + + + + + + + + + + {% set hold_types = invoices | map(attribute='hold_type') | unique | list %} + {% for hold in hold_types %} + + {% endfor %} + + + + + + {% else %} + + + + {% endif %} + +
    PMC NoVillage NameWork TypeInvoice DetailsInvoice DateInvoice NoBasic AmountDebitAfter Debit AmtGST (18%)AmountTDS (1%)SD (5%)On CommissionHydro Testing{{ hold }}GST SD (18%)Final Amount
    {{ invoice.PMC_No }}{{ invoice.Village_Name.capitalize() }}{{ invoice.Work_Type }}{{ invoice.Invoice_Details }}{{ invoice.Invoice_Date }}{{ invoice.Invoice_No }}{{ invoice.Basic_Amount }}{{ invoice.Debit_Amount }}{{ invoice.After_Debit_Amount }}{{ invoice.GST_Amount }}{{ invoice.Amount }}{{ invoice.TDS_Amount }}{{ invoice.SD_Amount }}{{ invoice.On_Commission }}{{ invoice.Hydro_Testing }} + {% if invoice.hold_type == hold %} + {{ invoice.hold_amount }} + {% else %} + 0.00 + {% endif %} + {{ invoice.GST_SD_Amount }}{{ invoice.Final_Amount }}
    Total{{total["sum_invo_basic_amt"]}}{{total["sum_invo_debit_amt"]}}{{total["sum_invo_after_debit_amt"]}}{{total["sum_invo_gst_amt"]}}{{total["sum_invo_amt"]}}{{total["sum_invo_tds_amt"]}}{{total["sum_invo_ds_amt"]}}{{total["sum_invo_on_commission"]}}{{total["sum_invo_hydro_test"]}}{{total["sum_invo_hold_amt"]}}{{total["sum_invo_gst_sd_amt"]}}{{total["sum_invo_final_amt"]}}
    No invoices found.
    + +

    Hold Release

    + + + + + + + + + + + + + {% if hold_release %} + {% for hold in hold_release %} + + + + + + + + + {% endfor %} + {% else %} + + + + {% endif %} + +
    PMC NoInvoice NoInvoice DetailsBasic AmountTotal AmountUTR
    {{ hold['PMC_No'] }}{{ hold['Invoice_No'] }}{{ hold['Invoice_Details'] }}{{ hold['Basic_Amount'] }}{{ hold['Total_Amount'] }}{{ hold['UTR'] }}
    No data present
    +
    + + + +

    Credit Details

    + + + + + + + + + + + + + + + + + + + {% if credit_note %} + {% for credit in credit_note %} + + + + + + + + + + + + + + + + {% endfor %} + + {% else %} + + + + {% endif %} + + + +
    PMC NoInvoice DetailsBasic AmountDebitAfter Debit AmtGST AmountAmountFinal AmountPayment AmountTotal AmountUTR
    {{ credit["PMC_No"] }}{{ credit["Invoice_Details"] }}{{ credit["Basic_Amount"] }}{{ credit["Debit_Amount"] }}{{ credit["After_Debit_Amount"] }}{{ credit["GST_Amount"] }}{{ credit["Amount"] }}{{ credit["Final_Amount"] }}{{ credit["Payment_Amount"] }}{{ credit["Total_Amount"] }}{{ credit["UTR"] }}
    No Credit note found.
    +

    GST Release Note Details

    + + + + + + + + + + + + + {% if gst_rel %} + {% for gst in gst_rel %} + + + + + + + {% endfor %} + + + + + + + {% else %} + + + + {% endif %} + +
    PMC NoInvoice NoBasic AmountFinal Amount
    {{ gst.pmc_no }}{{ gst.invoice_no }}{{ gst.basic_amount }}{{ gst.final_amount }}
    Total{{total["sum_gst_basic_amt"]}}{{total["sum_gst_final_amt"]}}
    No GST release found.
    + + +
    +

    Payment Details

    + + + + + + + + + + + + + {% if payments %} + {% for pay in payments %} + + + + + + + + + {% endfor %} + + + + + + + + + {% else %} + + + + + {% endif %} + + +
    PMC NoInvoice NoAmountTDS Amount @ 1% on BASIC AMOUNTTotal Amount PaidUTR
    {{ pay.pmc_no }}{{ pay.invoice_no }}{{ pay.Payment_Amount }}{{ pay.TDS_Payment_Amount }}{{ pay.Total_amount }}{{ pay.utr}}
    Total{{total["sum_pay_payment_amt"]}}{{total["sum_pay_tds_payment_amt"]}}{{total["sum_pay_total_amt"]}}
    No payment found.
    + + + Download Report +
    + +{% endblock %} \ No newline at end of file diff --git a/templates/unreleased_gst.html b/templates/unreleased_gst.html new file mode 100644 index 0000000..266a457 --- /dev/null +++ b/templates/unreleased_gst.html @@ -0,0 +1,35 @@ +{% extends "base.html" %} +{% block content %} +
    +

    🚫 GST Release Not Filled

    + + + + + + + + + + + + + + {% for row in data %} + + + + + + + + + {% endfor %} + +
    PMC NoInvoice NoInvoice DetailsGST_SD_Amount
    {{ row.PMC_No }}{{ row.Invoice_No }}{{ row.Invoice_Details }}{{ row.GST_SD_Amount }}
    + + {% if data|length == 0 %} +

    ✅ All invoices have GST releases.

    + {% endif %} +
    +{% endblock %} diff --git a/templates/uploadExcelFile.html b/templates/uploadExcelFile.html new file mode 100644 index 0000000..47c3a5b --- /dev/null +++ b/templates/uploadExcelFile.html @@ -0,0 +1,24 @@ +{% extends 'base.html' %} +{% block content %} + + + + Upload Excel File + + + + + +

    Upload Excel File

    +
    +
    + +

    + +
    +
    + + + +{% endblock %} diff --git a/templates/work_order_report.html b/templates/work_order_report.html new file mode 100644 index 0000000..80935e9 --- /dev/null +++ b/templates/work_order_report.html @@ -0,0 +1,209 @@ +{% extends 'base.html' %} +{% block content %} + + Work Order Report + + + + + +

    Work Order Report

    + +
    +
    + + +
    + +
    + + +
    + + + +
    + + + + + + + + + + + + + + + + + + + + + + + + + + +{% endblock %} diff --git a/unusedCode/unused.py b/unusedCode/unused.py new file mode 100644 index 0000000..e7c9245 --- /dev/null +++ b/unusedCode/unused.py @@ -0,0 +1,482 @@ + +logging.basicConfig(level=logging.DEBUG) + + +@app.route('/add_purchase_order', methods=['GET', 'POST']) +def add_purchase_order(): + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True) + + if request.method == 'POST': + # Fetch form fields + purchase_date = request.form.get('purchase_date') + supplier_name = request.form.get('supplier_name') + purchase_order_no = request.form.get('purchase_order_no') + item_name = request.form.get('item_name') + quantity = request.form.get('quantity') + unit = request.form.get('unit') + rate = request.form.get('rate') + amount = request.form.get('amount') + GST_Amount = request.form.get('GST_Amount') + TDS = request.form.get('TDS') + final_amount = request.form.get('final_amount') + LogHelper.log_action("Add purchase order", f"User {current_user.id} Added puirchase Order'{ purchase_order_no}'") + # Insert into database + insert_query = """ + INSERT INTO purchase_order ( + purchase_date, supplier_name, purchase_order_no, item_name, quantity, unit, rate, amount, + GST_Amount, TDS, final_amount + ) VALUES (%s, %s, %s, %s, %s, %s, %s, %s, %s, %s, %s) + """ + cursor.execute(insert_query, ( + purchase_date, supplier_name, purchase_order_no, item_name, quantity, unit, rate, amount, + GST_Amount, TDS, final_amount + )) + connection.commit() + + # ✅ Always fetch updated data + cursor.execute("SELECT * FROM purchase_order") + purchases = cursor.fetchall() + + cursor.close() + connection.close() + + return render_template('add_purchase_order.html', purchases=purchases) + + +# Show all purchases +@app.route('/purchase_orders') +def show_purchase_orders(): + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True) + cursor.execute("SELECT * FROM purchase_order") + purchases = cursor.fetchall() + cursor.close() + connection.close() + return render_template('add_purchase_order.html', purchases=purchases) + + +# Delete purchase order +@app.route('/delete_purchase/', methods=['POST']) +def delete_purchase(id): + connection = config.get_db_connection() + cursor = connection.cursor() + cursor.execute("DELETE FROM purchase_order WHERE purchase_id = %s", (id,)) + connection.commit() + cursor.close() + connection.close() + LogHelper.log_action("Delete purchase order", f"User {current_user.id} Deleted puirchase Order'{ id}'") + return render_template(('add_purchase_order.html')) + + +# Edit purchase order (form + update logic) +@app.route('/update_purchase/', methods=['GET', 'POST']) +def update_purchase(id): + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True) + + if request.method == 'POST': + # ✅ Form submitted - update all fields + data = request.form + cursor.execute(""" + UPDATE purchase_order + SET purchase_date = %s, + supplier_name = %s, + purchase_order_no = %s, + item_name = %s, + quantity = %s, + unit = %s, + rate = %s, + amount = %s, + GST_Amount = %s, + TDS = %s, + final_amount = %s + WHERE purchase_id = %s + """, ( + data['purchase_date'], data['supplier_name'], data['purchase_order_no'], data['item_name'], + data['quantity'], + data['unit'], data['rate'], data['amount'], data['GST_Amount'], data['TDS'], data['final_amount'], + id + )) + connection.commit() + cursor.close() + connection.close() + LogHelper.log_action("Delete purchase order", f"User {current_user.id} Deleted puirchase Order'{ id}'") + return redirect(url_for('show_purchase_orders')) + + # Show edit form + cursor.execute("SELECT * FROM purchase_order WHERE purchase_id = %s", (id,)) + purchase = cursor.fetchone() + cursor.close() + connection.close() + return render_template('edit_purchase.html', purchase=purchase) + + +# SHOW all GRNs + ADD form +@app.route('/grn', methods=['GET']) +def grn_page(): + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True) + + # Fetch purchase orders for dropdown + cursor.execute("SELECT purchase_id, supplier_name FROM purchase_order") + purchase_orders = cursor.fetchall() + + # Fetch all GRNs to display + cursor.execute("SELECT * FROM goods_receive_note") + grns = cursor.fetchall() + print(grns) + cursor.close() + connection.close() + + # Render the template with both datasets + return render_template('grn_form.html', purchase_orders=purchase_orders, grns=grns) + + +# ADD new GRN +@app.route('/add_grn', methods=['POST', 'GET']) +def add_grn(): + data = request.form + connection = config.get_db_connection() + cursor = connection.cursor() + query = """ + INSERT INTO goods_receive_note + (grn_date, purchase_id, supplier_name, item_description, received_quantity, unit, rate, amount, remarks) + VALUES (%s, %s, %s, %s, %s, %s, %s, %s, %s) \ + """ + cursor.execute(query, ( + data.get('grn_date'), + data.get('purchase_id'), + data.get('supplier_name'), + data.get('item_description'), + data.get('received_quantity'), + data.get('unit'), + data.get('rate'), + data.get('amount'), + data.get('remarks') + )) + connection.commit() + + cursor.execute("SELECT * FROM goods_receive_note") + grns = cursor.fetchall() + print(grns) + query = "select * from purchase_order" + cursor.execute(query) + purchase_orders = cursor.fetchall() + cursor.close() + connection.close() + return render_template('grn_form.html', purchase_orders=purchase_orders, grns=grns) + + +# UPDATE GRN +@app.route('/update_grn/', methods=['GET', 'POST']) +def update_grn(grn_id): + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True) + + if request.method == 'POST': + data = request.form + query = """ + UPDATE goods_receive_note + SET grn_date=%s, purchase_id=%s, supplier_name=%s, item_description=%s, + received_quantity=%s, unit=%s, rate=%s, amount=%s, remarks=%s + WHERE grn_id=%s + """ + cursor.execute(query, ( + data['grn_date'], data['purchase_id'], data['supplier_name'], + data['item_description'], data['received_quantity'], data['unit'], + data['rate'], data['amount'], data['remarks'], grn_id + )) + connection.commit() + cursor.close() + connection.close() + return redirect(url_for('grns')) + + cursor.execute("SELECT * FROM goods_receive_note WHERE grn_id = %s", (grn_id,)) + grn = cursor.fetchone() + cursor.close() + connection.close() + return render_template("edit_grn.html", grn=grn) + + +# DELETE GRN +@app.route('/delete_grn/', methods=['POST']) +def delete_grn(grn_id): + connection = config.get_db_connection() + cursor = connection.cursor() + cursor.execute("DELETE FROM goods_receive_note WHERE grn_id = %s", (grn_id,)) + connection.commit() + cursor.close() + connection.close() + return render_template("grn_form.html") + + +@app.route('/work_order_report', methods=['GET']) +def work_order_report(): + return render_template('work_order_report.html') + + + + +# ✅ Vendor Name Search (for Select2) +@app.route('/get_vendor_names') +def get_vendor_names(): + query = request.args.get('q', '') + connection = config.get_db_connection() + cursor = connection.cursor() + cursor.execute("SELECT DISTINCT vendor_name FROM work_order WHERE vendor_name LIKE %s", (f"%{query}%",)) + vendors = [row[0] for row in cursor.fetchall()] + cursor.close() + connection.close() + return jsonify(vendors) + + +# ✅ Work Order Number Search (with or without vendor) +@app.route('/get_work_order_numbers') +def get_work_order_numbers(): + vendor = request.args.get('vendor_name', '') + query = request.args.get('q', '') + connection = config.get_db_connection() + cursor = connection.cursor() + + if vendor: + cursor.execute( + "SELECT DISTINCT work_order_number FROM work_order WHERE vendor_name = %s AND work_order_number LIKE %s", + (vendor, f"%{query}%")) + else: + cursor.execute("SELECT DISTINCT work_order_number FROM work_order WHERE work_order_number LIKE %s", + (f"%{query}%",)) + + orders = [row[0] for row in cursor.fetchall()] + cursor.close() + connection.close() + return jsonify(orders) + + +# ✅ Get Work Order Data (Filtered) +@app.route('/get_work_order_data') +def get_work_order_data(): + vendor = request.args.get('vendor_name') + order_number = request.args.get('work_order_number') + + query = "SELECT * FROM work_order WHERE 1=1" + params = [] + + if vendor: + query += " AND vendor_name = %s" + params.append(vendor) + if order_number: + query += " AND work_order_number = %s" + params.append(order_number) + + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True) + cursor.execute(query, tuple(params)) + data = cursor.fetchall() + cursor.close() + connection.close() + return jsonify(data) + + + + +@app.route('/download_work_order_report') +def download_work_order_report(): + vendor_name = request.args.get('vendor_name') + work_order_number = request.args.get('work_order_number') + + if work_order_number == "null": + work_order_number = None + + query = "SELECT * FROM work_order WHERE 1=1" + params = [] + + if vendor_name: + query += " AND vendor_name = %s" + params.append(vendor_name) + if work_order_number: + query += " AND work_order_number = %s" + params.append(work_order_number) + + conn = config.get_db_connection() + cursor = conn.cursor(dictionary=True) + cursor.execute(query, tuple(params)) + data = cursor.fetchall() + cursor.close() + conn.close() + + if not data: + return "No data found for the selected filters", 404 + + # Convert to DataFrame + df = pd.DataFrame(data) + + output_path = 'static/downloads/work_order_report.xlsx' + os.makedirs(os.path.dirname(output_path), exist_ok=True) + df.to_excel(output_path, index=False, startrow=2) # Leave space for heading + + # Load workbook for styling + wb = load_workbook(output_path) + ws = wb.active + + # Add a merged title + title = "Work Order Report" + ws.merge_cells(start_row=1, start_column=1, end_row=1, end_column=len(df.columns)) + title_cell = ws.cell(row=1, column=1) + title_cell.value = title + title_cell.font = Font(size=14, bold=True, color="1F4E78") + title_cell.alignment = Alignment(horizontal="center", vertical="center") + + # Style header row (row 3 because data starts from row 3) + header_font = Font(bold=True) + for col_num, column_name in enumerate(df.columns, 1): + cell = ws.cell(row=3, column=col_num) + cell.font = header_font + cell.alignment = Alignment(horizontal="center", vertical="center") + # Optional: adjust column width + max_length = max(len(str(column_name)), 12) + ws.column_dimensions[get_column_letter(col_num)].width = max_length + 2 + + wb.save(output_path) + return send_file(output_path, as_attachment=True) + + +@app.route('/purchase_order_report', methods=['GET']) +def purchase_order_report(): + return render_template('purchase_order_report.html') + + +@app.route('/get_supplier_names') +def get_supplier_names(): + query = request.args.get('q', '') # Get the search term from Select2 + connection = config.get_db_connection() + cursor = connection.cursor() + + # Fetch distinct supplier names that match the search query + cursor.execute( + "SELECT supplier_name FROM purchase_order WHERE supplier_name LIKE %s", + (f"%{query}%",) + ) + suppliers = [row[0] for row in cursor.fetchall()] + + cursor.close() + connection.close() + return jsonify(suppliers) + + +@app.route('/get_purchase_order_numbers') +def get_purchase_order_numbers(): + supplier = request.args.get('supplier_name', '') + query = request.args.get('q', '') + + connection = config.get_db_connection() + cursor = connection.cursor() + + if supplier: + cursor.execute(""" + SELECT purchase_order_no + FROM purchase_order + WHERE supplier_name = %s AND purchase_order_no LIKE %s + """, (supplier, f"%{query}%")) + else: + cursor.execute(""" + SELECT purchase_order_no + FROM purchase_order + WHERE purchase_order_no LIKE %s + """, (f"%{query}%",)) + + orders = [row[0] for row in cursor.fetchall()] + cursor.close() + connection.close() + return jsonify(orders) + + +@app.route('/get_purchase_order_data') +def get_purchase_order_data(): + supplier = request.args.get('supplier_name') + order_no = request.args.get('purchase_order_no') + + query = "SELECT * FROM purchase_order WHERE 1=1" + params = [] + + if supplier: + query += " AND supplier_name = %s" + params.append(supplier) + if order_no: + query += " AND purchase_order_no = %s" + params.append(order_no) + + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True) + cursor.execute(query, tuple(params)) + data = cursor.fetchall() + cursor.close() + connection.close() + return jsonify(data) + + +@app.route('/download_purchase_order_report') +def download_purchase_order_report(): + supplier_name = request.args.get('supplier_name') + purchase_order_no = request.args.get('purchase_order_no') + + if purchase_order_no == "null": + purchase_order_no = None + LogHelper.log_action("Download purchase order", f"User {current_user.id} Download puirchase Order'{ purchase_order_no}'") + query = "SELECT * FROM purchase_order WHERE 1=1" + params = [] + + if supplier_name: + query += " AND supplier_name = %s" + params.append(supplier_name) + if purchase_order_no: + query += " AND purchase_order_no = %s" + params.append(purchase_order_no) + + conn = config.get_db_connection() + cursor = conn.cursor(dictionary=True) + cursor.execute(query, tuple(params)) + data = cursor.fetchall() + cursor.close() + conn.close() + + if not data: + return "No data found for the selected filters", 404 + + # Convert to DataFrame + df = pd.DataFrame(data) + + output_path = 'static/downloads/purchase_order_report.xlsx' + os.makedirs(os.path.dirname(output_path), exist_ok=True) + + df.to_excel(output_path, index=False, startrow=2) # Reserve space for heading + + # Load workbook for styling + wb = load_workbook(output_path) + ws = wb.active + + # Add a merged title + title = "Purchase Order Report" + ws.merge_cells(start_row=1, start_column=1, end_row=1, end_column=len(df.columns)) + title_cell = ws.cell(row=1, column=1) + title_cell.value = title + title_cell.font = Font(size=14, bold=True, color="1F4E78") + title_cell.alignment = Alignment(horizontal="center", vertical="center") + + # Style header row (row 3 because data starts from row 3) + header_font = Font(bold=True) + for col_num, column_name in enumerate(df.columns, 1): + cell = ws.cell(row=3, column=col_num) + cell.font = header_font + cell.alignment = Alignment(horizontal="center", vertical="center") + max_length = max(len(str(column_name)), 12) + ws.column_dimensions[get_column_letter(col_num)].width = max_length + 2 + + wb.save(output_path) + return send_file(output_path, as_attachment=True) + + +if __name__ == '__main__': + app.run(host='0.0.0.0', port=5000, debug=True) + + diff --git a/unusedCode/workOrder.py b/unusedCode/workOrder.py new file mode 100644 index 0000000..2623bb9 --- /dev/null +++ b/unusedCode/workOrder.py @@ -0,0 +1,128 @@ +# -- end hold types controlller -------------------- + +# Route to display the HTML form +@app.route('/add_work_order', methods=['GET']) +def add_work_order(): + # Add database connection + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True) + + cursor.execute("SELECT Contractor_id, Contractor_Name FROM subcontractors") # Adjust table/column names as needed + subcontractor = cursor.fetchall() + + cursor.close() + connection.close() + + return render_template('add_work_order.html', subcontractor=subcontractor) # This is your HTML form page + + +# Route to handle form submission (from action="/submit_work_order") +@app.route('/submit_work_order', methods=['POST', 'GET']) +def submit_work_order(): + vendor_name = request.form['vendor_name'] + work_order_type = request.form['work_order_type'] + work_order_amount = request.form['work_order_amount'] + boq_amount = request.form['boq_amount'] + work_done_percentage = request.form['work_done_percentage'] + work_order_number = request.form['work_order_number'] + gst_amount = request.form['gst_amount'] + tds_amount = request.form['tds_amount'] + security_deposite = request.form['security_deposite'] + sd_against_gst = request.form['sd_against_gst'] + final_total = request.form['final_total'] + tds_of_gst = request.form['tds_of_gst'] + LogHelper.log_action("Submit Work Order", f"User {current_user.id} Submit Work Order'{ work_order_type}'") + # print("Good Morning How are U") + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True) + # print("Good morning and how are you") + insert_query = """ + INSERT INTO work_order + (vendor_name, work_order_type, work_order_amount, boq_amount, work_done_percentage,work_order_number,gst_amount,tds_amount + ,security_deposit,sd_against_gst,final_total,tds_of_gst) + VALUES (%s, %s, %s, %s, %s,%s,%s,%s,%s,%s,%s,%s) + """ + cursor.execute(insert_query, + (vendor_name, work_order_type, work_order_amount, boq_amount, work_done_percentage, work_order_number + , gst_amount, tds_amount, security_deposite, sd_against_gst, final_total, tds_of_gst)) + connection.commit() + + # ✅ Fetch all data after insert + select_query = "SELECT * FROM work_order" + cursor.execute(select_query) + wo = cursor.fetchall() + # print("The Work order data is ",wo) + print("The data from work order ", wo) # should now print the data properly + cursor.execute("SELECT Contractor_id, Contractor_Name FROM subcontractors") # Adjust table/column names as needed + subcontractor = cursor.fetchall() + cursor.close() + connection.close() + + return render_template('add_work_order.html', work_order_type=work_order_type, wo=wo, subcontractor=subcontractor) + + +@app.route('/delete_work_order/', methods=['POST']) +def delete_work_order(id): + connection = config.get_db_connection() + cursor = connection.cursor() + cursor.execute("DELETE FROM work_order WHERE work_order_id = %s", (id,)) + connection.commit() + cursor.close() + connection.close() + LogHelper.log_action("delete Work Order", f"User {current_user.id} delete Work Order'{ id}'") + return jsonify({'success': True}) + + +@app.route('/update_work_order/', methods=['GET', 'POST']) +def update_work_order(id): + connection = config.get_db_connection() + cursor = connection.cursor(dictionary=True) + + if request.method == 'POST': + work_order_type = request.form['work_order_type'] + work_order_amount = request.form['work_order_amount'] + boq_amount = request.form['boq_amount'] + work_done_percentage = request.form['work_done_percentage'] + work_order_number = request.form['work_order_number'] + gst_amount = request.form['gst_amount'] + tds_amount = request.form['tds_amount'] + security_deposite = request.form['security_deposite'] + sd_against_gst = request.form['sd_against_gst'] + final_amount = request.form['final_amount'] + tds_of_gst = request.form['tds_of_gst'] + update_query = """ + UPDATE work_order + SET work_order_type = %s, + work_order_amount = %s, + boq_amount = %s, + work_done_percentage = %s, + work_order_number= %s, + gst_amount = %s, + tds_amount= %s, + security_deposite= %s, + sd_against_gst=%s, + final_amount= %s, + tds_of_gst=%s + WHERE work_order_id = %s + """ + cursor.execute(update_query, ( + work_order_type, work_order_amount, boq_amount, work_done_percentage, work_order_number, gst_amount, + tds_amount, security_deposite, sd_against_gst, final_amount, tds_of_gst, id)) + connection.commit() + cursor.close() + connection.close() + + # If GET request: fetch the existing record + cursor.execute("SELECT * FROM work_order WHERE work_order_id = %s", (id,)) + work_order = cursor.fetchone() + cursor.close() + connection.close() + return render_template('update_work_order.html', work_order=work_order) + + +# Optional: Route to show a success message +@app.route('/success') +def success(): + return "Work Order Submitted Successfully!" + +