Rename controller namespaces: drop V1 prefix, clarify roles

V1/Admin     -> Company       (company-scoped controllers)
V1/SuperAdmin -> Admin        (platform-wide admin controllers)
V1/Customer  -> CustomerPortal (customer-facing portal)
V1/Installation -> Setup      (installation wizard)
V1/PDF       -> Pdf           (consistent casing)
V1/Modules   -> Modules       (drop V1 prefix)
V1/Webhook   -> Webhook       (drop V1 prefix)

The V1 prefix served no purpose - API versioning is in the route prefix
(/api/v1/), not the controller namespace. "Admin" was misleading for
company-scoped controllers. "SuperAdmin" is now simply "Admin" for
platform administration.
This commit is contained in:
Darko Gjorgjijoski
2026-04-03 19:15:20 +02:00
parent 0aaf0419c3
commit 64c481e963
129 changed files with 236 additions and 236 deletions

View File

@@ -0,0 +1,53 @@
<?php
namespace App\Http\Controllers\Company\Settings;
use App\Http\Controllers\Controller;
use App\Http\Requests\CompanyLogoRequest;
use App\Http\Requests\CompanyRequest;
use App\Http\Resources\CompanyResource;
use App\Models\Company;
class CompanyController extends Controller
{
public function updateCompany(CompanyRequest $request)
{
$company = Company::find($request->header('company'));
$this->authorize('manage company', $company);
$company->update($request->getCompanyPayload());
$company->address()->updateOrCreate(['company_id' => $company->id], $request->address);
return new CompanyResource($company);
}
public function uploadCompanyLogo(CompanyLogoRequest $request)
{
$company = Company::find($request->header('company'));
$this->authorize('manage company', $company);
$data = json_decode($request->company_logo);
if (isset($request->is_company_logo_removed) && (bool) $request->is_company_logo_removed) {
$company->clearMediaCollection('logo');
}
if ($data) {
$company = Company::find($request->header('company'));
if ($company) {
$company->clearMediaCollection('logo');
$company->addMediaFromBase64($data->data)
->usingFileName($data->name)
->toMediaCollection('logo');
}
}
return response()->json([
'success' => true,
]);
}
}

View File

@@ -0,0 +1,27 @@
<?php
namespace App\Http\Controllers\Company\Settings;
use App\Http\Controllers\Controller;
use App\Models\Company;
use Illuminate\Http\Request;
use Illuminate\Http\Response;
class CompanyCurrencyCheckTransactionsController extends Controller
{
/**
* Handle the incoming request.
*
* @return Response
*/
public function __invoke(Request $request)
{
$company = Company::find($request->header('company'));
$this->authorize('manage company', $company);
return response()->json([
'has_transactions' => $company->hasTransactions(),
]);
}
}

View File

@@ -0,0 +1,155 @@
<?php
namespace App\Http\Controllers\Company\Settings;
use App\Http\Controllers\Controller;
use App\Mail\TestMail;
use App\Models\CompanySetting;
use App\Services\CompanyMailConfigService;
use Illuminate\Http\JsonResponse;
use Illuminate\Http\Request;
use Mail;
class CompanyMailConfigurationController extends Controller
{
public function getDefaultConfig(Request $request): JsonResponse
{
$mailConfig = [
'from_name' => config('mail.from.name'),
'from_mail' => config('mail.from.address'),
];
return response()->json($mailConfig);
}
public function getMailConfig(Request $request): JsonResponse
{
$companyId = $request->header('company');
$useCustom = CompanySetting::getSetting('use_custom_mail_config', $companyId) ?? 'NO';
$driver = CompanySetting::getSetting('company_mail_driver', $companyId) ?? '';
$data = [
'use_custom_mail_config' => $useCustom,
'mail_driver' => $driver,
'from_name' => CompanySetting::getSetting('company_from_name', $companyId) ?? '',
'from_mail' => CompanySetting::getSetting('company_from_mail', $companyId) ?? '',
];
switch ($driver) {
case 'smtp':
$data = array_merge($data, [
'mail_host' => CompanySetting::getSetting('company_mail_host', $companyId) ?? '',
'mail_port' => CompanySetting::getSetting('company_mail_port', $companyId) ?? '',
'mail_username' => CompanySetting::getSetting('company_mail_username', $companyId) ?? '',
'mail_password' => CompanySetting::getSetting('company_mail_password', $companyId) ?? '',
'mail_encryption' => CompanySetting::getSetting('company_mail_encryption', $companyId) ?? 'none',
'mail_scheme' => CompanySetting::getSetting('company_mail_scheme', $companyId) ?? '',
'mail_url' => CompanySetting::getSetting('company_mail_url', $companyId) ?? '',
'mail_timeout' => CompanySetting::getSetting('company_mail_timeout', $companyId) ?? '',
'mail_local_domain' => CompanySetting::getSetting('company_mail_local_domain', $companyId) ?? '',
]);
break;
case 'mailgun':
$data = array_merge($data, [
'mail_mailgun_domain' => CompanySetting::getSetting('company_mail_mailgun_domain', $companyId) ?? '',
'mail_mailgun_secret' => CompanySetting::getSetting('company_mail_mailgun_secret', $companyId) ?? '',
'mail_mailgun_endpoint' => CompanySetting::getSetting('company_mail_mailgun_endpoint', $companyId) ?? 'api.mailgun.net',
'mail_mailgun_scheme' => CompanySetting::getSetting('company_mail_mailgun_scheme', $companyId) ?? 'https',
]);
break;
case 'ses':
$data = array_merge($data, [
'mail_ses_key' => CompanySetting::getSetting('company_mail_ses_key', $companyId) ?? '',
'mail_ses_secret' => CompanySetting::getSetting('company_mail_ses_secret', $companyId) ?? '',
'mail_ses_region' => CompanySetting::getSetting('company_mail_ses_region', $companyId) ?? 'us-east-1',
]);
break;
case 'sendmail':
$data = array_merge($data, [
'mail_sendmail_path' => CompanySetting::getSetting('company_mail_sendmail_path', $companyId) ?? '/usr/sbin/sendmail -bs -i',
]);
break;
}
return response()->json($data);
}
public function saveMailConfig(Request $request): JsonResponse
{
$this->authorize('owner only');
$companyId = $request->header('company');
$driver = $request->get('mail_driver', '');
$settings = [
'use_custom_mail_config' => $request->get('use_custom_mail_config', 'NO'),
'company_mail_driver' => $driver,
'company_from_name' => $request->get('from_name', ''),
'company_from_mail' => $request->get('from_mail', ''),
];
switch ($driver) {
case 'smtp':
$settings = array_merge($settings, [
'company_mail_host' => $request->get('mail_host', ''),
'company_mail_port' => $request->get('mail_port', ''),
'company_mail_username' => $request->get('mail_username', ''),
'company_mail_password' => $request->get('mail_password', ''),
'company_mail_encryption' => $request->get('mail_encryption', 'none'),
'company_mail_scheme' => $request->get('mail_scheme', ''),
'company_mail_url' => $request->get('mail_url', ''),
'company_mail_timeout' => $request->get('mail_timeout', ''),
'company_mail_local_domain' => $request->get('mail_local_domain', ''),
]);
break;
case 'mailgun':
$settings = array_merge($settings, [
'company_mail_mailgun_domain' => $request->get('mail_mailgun_domain', ''),
'company_mail_mailgun_secret' => $request->get('mail_mailgun_secret', ''),
'company_mail_mailgun_endpoint' => $request->get('mail_mailgun_endpoint', 'api.mailgun.net'),
'company_mail_mailgun_scheme' => $request->get('mail_mailgun_scheme', 'https'),
]);
break;
case 'ses':
$settings = array_merge($settings, [
'company_mail_ses_key' => $request->get('mail_ses_key', ''),
'company_mail_ses_secret' => $request->get('mail_ses_secret', ''),
'company_mail_ses_region' => $request->get('mail_ses_region', 'us-east-1'),
]);
break;
case 'sendmail':
$settings = array_merge($settings, [
'company_mail_sendmail_path' => $request->get('mail_sendmail_path', '/usr/sbin/sendmail -bs -i'),
]);
break;
}
CompanySetting::setSettings($settings, $companyId);
return response()->json(['success' => true]);
}
public function testMailConfig(Request $request): JsonResponse
{
$this->authorize('owner only');
$this->validate($request, [
'to' => 'required|email',
'subject' => 'required',
'message' => 'required',
]);
CompanyMailConfigService::apply($request->header('company'));
Mail::to($request->to)->send(new TestMail($request->subject, $request->message));
return response()->json(['success' => true]);
}
}

View File

@@ -0,0 +1,46 @@
<?php
namespace App\Http\Controllers\Company\Settings;
use App\Http\Controllers\Controller;
use App\Http\Requests\GetSettingsRequest;
use App\Http\Requests\UpdateSettingsRequest;
use App\Models\Company;
use App\Models\CompanySetting;
use Illuminate\Http\JsonResponse;
use Illuminate\Support\Arr;
class CompanySettingsController extends Controller
{
public function show(GetSettingsRequest $request): JsonResponse
{
$settings = CompanySetting::getSettings((array) $request->settings, $request->header('company'));
return response()->json($settings);
}
public function update(UpdateSettingsRequest $request): JsonResponse
{
$company = Company::find($request->header('company'));
$this->authorize('manage company', $company);
$data = $request->settings;
if (
Arr::exists($data, 'currency') &&
(CompanySetting::getSetting('currency', $company->id) !== $data['currency']) &&
$company->hasTransactions()
) {
return response()->json([
'success' => false,
'message' => 'Cannot update company currency after transactions are created.',
]);
}
CompanySetting::setSettings($data, $request->header('company'));
return response()->json([
'success' => true,
]);
}
}

View File

@@ -0,0 +1,95 @@
<?php
namespace App\Http\Controllers\Company\Settings;
use App\Http\Controllers\Controller;
use App\Http\Requests\TaxTypeRequest;
use App\Http\Resources\TaxTypeResource;
use App\Models\TaxType;
use Illuminate\Http\Request;
use Illuminate\Http\Response;
class TaxTypesController extends Controller
{
/**
* Display a listing of the resource.
*
* @return Response
*/
public function index(Request $request)
{
$this->authorize('viewAny', TaxType::class);
$limit = $request->has('limit') ? $request->limit : 5;
$taxTypes = TaxType::applyFilters($request->all())
->where('type', TaxType::TYPE_GENERAL)
->whereCompany()
->latest()
->paginateData($limit);
return TaxTypeResource::collection($taxTypes);
}
/**
* Store a newly created resource in storage.
*
* @param Request $request
* @return Response
*/
public function store(TaxTypeRequest $request)
{
$this->authorize('create', TaxType::class);
$taxType = TaxType::create($request->getTaxTypePayload());
return new TaxTypeResource($taxType);
}
/**
* Display the specified resource.
*
* @return Response
*/
public function show(TaxType $taxType)
{
$this->authorize('view', $taxType);
return new TaxTypeResource($taxType);
}
/**
* Update the specified resource in storage.
*
* @param Request $request
* @return Response
*/
public function update(TaxTypeRequest $request, TaxType $taxType)
{
$this->authorize('update', $taxType);
$taxType->update($request->getTaxTypePayload());
return new TaxTypeResource($taxType);
}
/**
* Remove the specified resource from storage.
*
* @return Response
*/
public function destroy(TaxType $taxType)
{
$this->authorize('delete', $taxType);
if ($taxType->taxes() && $taxType->taxes()->count() > 0) {
return respondJson('taxes_attached', 'Taxes Attached.');
}
$taxType->delete();
return response()->json([
'success' => true,
]);
}
}

View File

@@ -0,0 +1,52 @@
<?php
namespace App\Http\Controllers\Company\Settings;
use App\Http\Controllers\Controller;
use App\Http\Requests\AvatarRequest;
use App\Http\Requests\ProfileRequest;
use App\Http\Resources\UserResource;
use Illuminate\Http\Request;
class UserProfileController extends Controller
{
public function show(Request $request)
{
return new UserResource($request->user());
}
public function update(ProfileRequest $request)
{
$user = $request->user();
$user->update($request->validated());
return new UserResource($user);
}
public function uploadAvatar(AvatarRequest $request)
{
$user = auth()->user();
if (isset($request->is_admin_avatar_removed) && (bool) $request->is_admin_avatar_removed) {
$user->clearMediaCollection('admin_avatar');
}
if ($user && $request->hasFile('admin_avatar')) {
$user->clearMediaCollection('admin_avatar');
$user->addMediaFromRequest('admin_avatar')
->toMediaCollection('admin_avatar');
}
if ($user && $request->has('avatar')) {
$data = json_decode($request->avatar);
$user->clearMediaCollection('admin_avatar');
$user->addMediaFromBase64($data->data)
->usingFileName($data->name)
->toMediaCollection('admin_avatar');
}
return new UserResource($user);
}
}

View File

@@ -0,0 +1,29 @@
<?php
namespace App\Http\Controllers\Company\Settings;
use App\Http\Controllers\Controller;
use App\Http\Requests\GetSettingsRequest;
use App\Http\Requests\UpdateSettingsRequest;
use Illuminate\Http\JsonResponse;
class UserSettingsController extends Controller
{
public function show(GetSettingsRequest $request): JsonResponse
{
$user = $request->user();
return response()->json($user->getSettings((array) $request->settings));
}
public function update(UpdateSettingsRequest $request): JsonResponse
{
$user = $request->user();
$user->setSettings($request->settings);
return response()->json([
'success' => true,
]);
}
}