commit 1a825ba46cf2a632b551b1e29ecb85bedfe79d35 Author: akashbhome Date: Fri Mar 27 14:17:21 2026 +0530 updated Payment reconcillation code 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 0000000..6b4e1de Binary files /dev/null and b/download/Contractor_Report_1.xlsx differ 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 0000000..82b0204 Binary files /dev/null and b/static/download/Contractor_Report_1.xlsx differ diff --git a/static/download/Contractor_Report_2.xlsx b/static/download/Contractor_Report_2.xlsx new file mode 100644 index 0000000..c38616c Binary files /dev/null and b/static/download/Contractor_Report_2.xlsx differ diff --git a/static/download/Contractor_Report_3.xlsx b/static/download/Contractor_Report_3.xlsx new file mode 100644 index 0000000..3232916 Binary files /dev/null and b/static/download/Contractor_Report_3.xlsx differ diff --git a/static/download/Contractor_Report_4.xlsx b/static/download/Contractor_Report_4.xlsx new file mode 100644 index 0000000..aab8bce Binary files /dev/null and b/static/download/Contractor_Report_4.xlsx differ diff --git a/static/images/icons/bin_red_icon.png b/static/images/icons/bin_red_icon.png new file mode 100644 index 0000000..ca80068 Binary files /dev/null and b/static/images/icons/bin_red_icon.png differ diff --git a/static/images/icons/pen_blue_icon.png b/static/images/icons/pen_blue_icon.png new file mode 100644 index 0000000..2f0bf5f Binary files /dev/null and b/static/images/icons/pen_blue_icon.png differ diff --git a/static/js/block.js b/static/js/block.js new file mode 100644 index 0000000..2448888 --- /dev/null +++ b/static/js/block.js @@ -0,0 +1,116 @@ +window.onload = function () { + document.getElementById('block_Name').focus(); +}; + +$(document).ready(function () { + + $("#block_Name").on("input", function () { + let blockName = $(this).val(); + let cleanedName = blockName.replace(/[^A-Za-z ]/g, ""); + $(this).val(cleanedName); + }); + + $("#block_Name, #district_Id").on("input change", function () { + + let blockName = $("#block_Name").val().trim(); + let districtId = $("#district_Id").val(); + + if (blockName === "" || districtId === "") { + $("#blockMessage").text("").css("color", ""); + $("#submitButton").prop("disabled", true); + return; + } + + $.ajax({ + url: "/check_block", + type: "POST", + contentType: "application/json", + data: JSON.stringify({ + block_Name: blockName, + district_Id: districtId + }), + success: function (response) { + + if (response.status === "available") { + $("#blockMessage").text(response.message).css("color", "green"); + $("#submitButton").prop("disabled", false); + } + }, + error: function (xhr) { + + if (xhr.status === 409) { + $("#blockMessage").text("Block already exists!").css("color", "red"); + $("#submitButton").prop("disabled", true); + } + else if (xhr.status === 400) { + $("#blockMessage").text("Invalid block name! Only letters allowed.").css("color", "red"); + $("#submitButton").prop("disabled", true); + } + } + }); + }); + + + $("#blockForm").on("submit", function (event) { + + event.preventDefault(); + + $.ajax({ + url: "/add_block", + type: "POST", + data: $(this).serialize(), + + success: function (response) { + alert(response.message); + location.reload(); + }, + + error: function (xhr) { + alert(xhr.responseJSON.message); + } + }); + }); + + + + // ✅ FETCH DISTRICTS WHEN STATE CHANGES + $('#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); + }, + + 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!" + +