import { auth }            from '@/auth';
import { prisma }          from '@/lib/db/prisma';
import { GLOBAL_LIVE_UPDATE_COMPANY_ID, publishLiveUpdate } from '@/lib/live-updates/server';
import { assertWarehouseModeTransition, ensureCompanyFallbackWarehouse, normalizeWarehouseMode } from '@/lib/warehouses/companyWarehouseMode';
import { successResponse, errorResponse } from '@/lib/utils/apiResponse';
import { normalizeCompanyPrintTemplateShape } from '@/lib/utils/companyPrintTemplates';
import { normalizeStockControlSettings, mergeStockControlSettingsIntoCompanySettings } from '@/lib/stock-control/settings';
import { Prisma }          from '@prisma/client';
import { z }               from 'zod';

const UpdateSchema = z.object({
  name:           z.string().min(1).max(100).optional(),
  description:    z.string().max(300).optional(),
  isActive:       z.boolean().optional(),
  address:        z.string().max(500).optional(),
  phone:          z.string().max(50).optional(),
  email:          z.string().email().optional().or(z.literal('')),
  printTemplates: z.any().optional(),
  externalCompanyId: z.string().max(120).optional().or(z.literal('')),
  jobSourceMode: z.enum(['HYBRID', 'EXTERNAL_ONLY', 'INTERNAL_ONLY']).optional(),
  customerSourceMode: z.enum(['HYBRID', 'EXTERNAL_ONLY', 'INTERNAL_ONLY']).optional(),
  supplierSourceMode: z.enum(['HYBRID', 'EXTERNAL_ONLY', 'INTERNAL_ONLY']).optional(),
  jobCostingSettings: z.any().optional(),
  operationalSettings: z
    .object({
      currencyCode: z.string().trim().min(3).max(3).optional(),
    })
    .optional(),
});

export async function GET(_: Request, { params }: { params: Promise<{ id: string }> }) {
  const session = await auth();
  if (!session?.user) return errorResponse('Unauthorized', 401);
  const { id } = await params;

  const company = await prisma.company.findUnique({
    where: { id },
    include: {
      stockFallbackWarehouse: {
        select: { id: true, name: true },
      },
    },
  });
  if (!company) return errorResponse('Company not found', 404);

  return successResponse(normalizeCompanyPrintTemplateShape(company as Record<string, unknown>));
}

export async function PUT(req: Request, { params }: { params: Promise<{ id: string }> }) {
  const session = await auth();
  if (!session?.user) return errorResponse('Unauthorized', 401);

  const isSA = session.user.isSuperAdmin ?? false;
  const perms = (session.user.permissions ?? []) as string[];
  const canManageSettings = isSA || perms.includes('settings.manage');

  const { id } = await params;

  const body   = await req.json();
  const parsed = UpdateSchema.safeParse(body);
  if (!parsed.success) return errorResponse(parsed.error.issues[0]?.message ?? 'Validation error', 422);

  // Non-SA can only update profile fields (address, phone, email) for their own company
  // SA can update anything including name, description, isActive
  if (!isSA) {
    // Check if trying to update restricted fields
    const restrictedFields = ['name', 'description', 'isActive'];
    const hasRestrictedFields = restrictedFields.some((field) => parsed.data[field as keyof typeof parsed.data] !== undefined);
    if (hasRestrictedFields || !canManageSettings) {
      return errorResponse('Forbidden', 403);
    }
    // Non-SA can only update their own company
    if (id !== session.user.activeCompanyId) {
      return errorResponse('Forbidden', 403);
    }
  }

  const update: Prisma.CompanyUpdateInput = {};
  if (isSA && parsed.data.name !== undefined) update.name = parsed.data.name;
  if (isSA && parsed.data.description !== undefined) update.description = parsed.data.description;
  if (isSA && parsed.data.isActive !== undefined) update.isActive = parsed.data.isActive;
  if (parsed.data.address !== undefined) update.address = parsed.data.address;
  if (parsed.data.phone !== undefined) update.phone = parsed.data.phone;
  if (parsed.data.email !== undefined) update.email = parsed.data.email;
  if (parsed.data.printTemplates !== undefined) update.printTemplates = parsed.data.printTemplates;
  if (isSA && parsed.data.externalCompanyId !== undefined) {
    update.externalCompanyId = parsed.data.externalCompanyId || null;
  }
  if (isSA && parsed.data.jobSourceMode !== undefined) {
    update.jobSourceMode = parsed.data.jobSourceMode;
  }
  if (isSA && parsed.data.customerSourceMode !== undefined) {
    update.customerSourceMode = parsed.data.customerSourceMode;
  }
  if (isSA && parsed.data.supplierSourceMode !== undefined) {
    update.supplierSourceMode = parsed.data.supplierSourceMode;
  }
  if (isSA && parsed.data.jobCostingSettings !== undefined) {
    update.jobCostingSettings = mergeStockControlSettingsIntoCompanySettings(
      parsed.data.jobCostingSettings,
      normalizeStockControlSettings(
        parsed.data.jobCostingSettings &&
          typeof parsed.data.jobCostingSettings === 'object' &&
          !Array.isArray(parsed.data.jobCostingSettings)
          ? (parsed.data.jobCostingSettings as { stockControl?: unknown }).stockControl
          : undefined
      )
    );
  }
  if (isSA) {
    update.warehouseMode = normalizeWarehouseMode(undefined);
  }

  const company = await prisma.$transaction(async (tx) => {
    const existing = await tx.company.findUniqueOrThrow({
      where: { id },
      select: { operationalSettings: true },
    });

    if (isSA && parsed.data.operationalSettings !== undefined) {
      const currentOperational =
        existing.operationalSettings &&
        typeof existing.operationalSettings === 'object' &&
        !Array.isArray(existing.operationalSettings)
          ? (existing.operationalSettings as Record<string, unknown>)
          : {};

      const nextOperationalSettings: Record<string, unknown> = {
        ...currentOperational,
        ...(parsed.data.operationalSettings.currencyCode
          ? { currencyCode: parsed.data.operationalSettings.currencyCode.toUpperCase() }
          : {}),
      };
      update.operationalSettings = nextOperationalSettings as Prisma.InputJsonValue;
    }

    if (isSA) {
      await assertWarehouseModeTransition(tx, id, normalizeWarehouseMode(undefined));
    }

    const updated = await tx.company.update({
      where: { id },
      data: update,
    });

    await ensureCompanyFallbackWarehouse(tx, id);

    return tx.company.findUniqueOrThrow({
      where: { id: updated.id },
      include: {
        stockFallbackWarehouse: {
          select: { id: true, name: true },
        },
      },
    });
  });

  if (!company) return errorResponse('Company not found', 404);
  publishLiveUpdate({
    companyId: GLOBAL_LIVE_UPDATE_COMPANY_ID,
    channel: 'admin',
    entity: 'company',
    action: 'updated',
  });
  return successResponse(normalizeCompanyPrintTemplateShape(company as Record<string, unknown>));
}
