123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130 |
- <?php
-
- namespace App\Services;
-
- use App\Enums\Accounting\AccountType;
- use App\Enums\Banking\BankAccountType;
- use App\Models\Accounting\AccountSubtype;
- use App\Models\Accounting\Adjustment;
- use App\Models\Banking\BankAccount;
- use App\Models\Company;
- use App\Utilities\Currency\CurrencyAccessor;
- use Exception;
-
- class ChartOfAccountsService
- {
- public function createChartOfAccounts(Company $company): void
- {
- $chartOfAccounts = config('chart-of-accounts.default');
-
- foreach ($chartOfAccounts as $type => $subtypes) {
- foreach ($subtypes as $subtypeName => $subtypeConfig) {
- $subtype = $company->accountSubtypes()
- ->createQuietly([
- 'multi_currency' => $subtypeConfig['multi_currency'] ?? false,
- 'inverse_cash_flow' => $subtypeConfig['inverse_cash_flow'] ?? false,
- 'category' => AccountType::from($type)->getCategory()->value,
- 'type' => $type,
- 'name' => $subtypeName,
- 'description' => $subtypeConfig['description'] ?? 'No description available.',
- ]);
-
- try {
- $this->createDefaultAccounts($company, $subtype, $subtypeConfig);
- } catch (Exception $e) {
- // Log the error
- logger()->alert('Failed to create a company with its defaults, blocking critical business functionality.', [
- 'error' => $e->getMessage(),
- 'userId' => $company->owner->id,
- 'companyId' => $company->id,
- ]);
-
- throw $e;
- }
- }
- }
- }
-
- private function createDefaultAccounts(Company $company, AccountSubtype $subtype, array $subtypeConfig): void
- {
- if (isset($subtypeConfig['accounts']) && is_array($subtypeConfig['accounts'])) {
- $baseCode = $subtypeConfig['base_code'];
- $defaultCurrencyCode = CurrencyAccessor::getDefaultCurrency();
-
- if (empty($defaultCurrencyCode)) {
- throw new Exception('No default currency available for creating accounts.');
- }
-
- foreach ($subtypeConfig['accounts'] as $accountName => $accountDetails) {
- // Create the Account without directly setting bank_account_id
- $account = $company->accounts()->createQuietly([
- 'subtype_id' => $subtype->id,
- 'category' => $subtype->type->getCategory()->value,
- 'type' => $subtype->type->value,
- 'code' => $baseCode++,
- 'name' => $accountName,
- 'currency_code' => $defaultCurrencyCode,
- 'description' => $accountDetails['description'] ?? 'No description available.',
- 'default' => true,
- 'created_by' => $company->owner->id,
- 'updated_by' => $company->owner->id,
- ]);
-
- // Check if we need to create a BankAccount for this Account
- if ($subtypeConfig['multi_currency'] && isset($subtypeConfig['bank_account_type'])) {
- $bankAccount = $this->createBankAccountForMultiCurrency($company, $subtypeConfig['bank_account_type']);
-
- // Associate the BankAccount with the Account
- $bankAccount->account()->associate($account);
- $bankAccount->saveQuietly();
- }
-
- if (isset($subtypeConfig['adjustment_category'], $subtypeConfig['adjustment_type'])) {
- $adjustment = $this->createAdjustmentForAccount($company, $subtypeConfig['adjustment_category'], $subtypeConfig['adjustment_type']);
-
- // Associate the Adjustment with the Account
- $adjustment->account()->associate($account);
- $adjustment->saveQuietly();
- }
- }
- }
- }
-
- private function createBankAccountForMultiCurrency(Company $company, string $bankAccountType): BankAccount
- {
- $noDefaultBankAccount = $company->bankAccounts()->where('enabled', true)->doesntExist();
-
- return $company->bankAccounts()->createQuietly([
- 'type' => BankAccountType::from($bankAccountType) ?? BankAccountType::Other,
- 'enabled' => $noDefaultBankAccount,
- 'created_by' => $company->owner->id,
- 'updated_by' => $company->owner->id,
- ]);
- }
-
- private function createAdjustmentForAccount(Company $company, string $category, string $type): Adjustment
- {
- $noDefaultAdjustmentType = $company->adjustments()->where('category', $category)
- ->where('type', $type)
- ->where('enabled', true)
- ->doesntExist();
-
- $defaultRate = match ([$category, $type]) {
- ['tax', 'sales'] => 8, // Default 8% for Sales Tax
- ['tax', 'purchase'] => 8, // Default 8% for Purchase Tax
- ['discount', 'sales'] => 5, // Default 5% for Sales Discount
- ['discount', 'purchase'] => 5, // Default 5% for Purchase Discount
- default => 0, // Default to 0 if unspecified
- };
-
- return $company->adjustments()->createQuietly([
- 'category' => $category,
- 'type' => $type,
- 'rate' => $defaultRate,
- 'computation' => 'percentage',
- 'enabled' => $noDefaultAdjustmentType,
- 'created_by' => $company->owner->id,
- 'updated_by' => $company->owner->id,
- ]);
- }
- }
|