Files
InvoiceShelf/tests/Feature/Company/Modules/CompanyModulesIndexTest.php
Darko Gjorgjijoski 93b04a0c2a test(modules): integration tests for company surfaces and stub generator
End-to-end coverage for the new module APIs and the custom module:make
stubs shipped from invoiceshelf/modules. Each test file is hermetic — uses
\InvoiceShelf\Modules\Registry::flush() in setup/teardown to prevent
cross-test contamination, and ModuleMakeStubTest cleans up generated test
artifacts (the throwaway scaffold directory and the storage statuses entry).

- CompanyModulesIndexTest: 4 tests covering only-enabled-modules filter,
  has_settings flag computed against the real Registry, menu inclusion, and
  the empty-state response.

- ModuleSettingsControllerTest: 7 tests covering 404 for unregistered slug,
  show schema + defaults round-trip, persistence with the
  module.{slug}.{key} prefix, missing-required-field rejection, unknown-key
  silent-drop, update 404, and per-company isolation (the load-bearing
  multi-tenancy guarantee).

- BootstrapModuleMenuTest: 3 tests covering Registry-driven module_menu
  population on the company-context bootstrap branch, the empty default
  when nothing is registered, and the absence of module_menu on the
  super-admin-mode branch.

- ModuleMakeStubTest: 3 tests that actually run
  Artisan::call('module:make', ['name' => ['ScaffoldProbe']]) against a
  throwaway module name and assert the generated ServiceProvider contains
  use InvoiceShelf\Modules\Registry, the generated composer.json requires
  invoiceshelf/modules: ^3.0, and starter lang/en/{menu,settings}.php exist.
  Validates that the custom stubs shipped from the package are picked up
  via Stub::setBasePath().
2026-04-09 00:30:24 +02:00

105 lines
2.7 KiB
PHP

<?php
use App\Models\Module;
use App\Models\User;
use Illuminate\Support\Facades\Artisan;
use InvoiceShelf\Modules\Registry;
use Laravel\Sanctum\Sanctum;
use function Pest\Laravel\getJson;
beforeEach(function () {
Artisan::call('db:seed', ['--class' => 'DatabaseSeeder', '--force' => true]);
Artisan::call('db:seed', ['--class' => 'DemoSeeder', '--force' => true]);
$user = User::find(1);
$this->withHeaders([
'company' => $user->companies()->first()->id,
]);
Sanctum::actingAs($user, ['*']);
Registry::flush();
});
afterEach(function () {
Registry::flush();
});
test('returns only enabled modules', function () {
Module::create([
'name' => 'sales-tax-us',
'version' => '1.0.0',
'installed' => true,
'enabled' => true,
]);
Module::create([
'name' => 'archived-module',
'version' => '0.5.0',
'installed' => true,
'enabled' => false,
]);
$response = getJson('api/v1/company-modules')->assertOk();
$response->assertJsonCount(1, 'data');
$response->assertJsonPath('data.0.slug', 'sales-tax-us');
});
test('reports has_settings flag based on Registry', function () {
Module::create([
'name' => 'with-settings',
'version' => '1.0.0',
'installed' => true,
'enabled' => true,
]);
Module::create([
'name' => 'without-settings',
'version' => '1.0.0',
'installed' => true,
'enabled' => true,
]);
Registry::registerSettings('with-settings', [
'sections' => [
['title' => 'general', 'fields' => [
['key' => 'foo', 'type' => 'text'],
]],
],
]);
$response = getJson('api/v1/company-modules')->assertOk();
$rows = collect($response->json('data'))->keyBy('slug');
expect($rows['with-settings']['has_settings'])->toBeTrue();
expect($rows['without-settings']['has_settings'])->toBeFalse();
});
test('includes registered menu entry for active modules', function () {
Module::create([
'name' => 'menu-module',
'version' => '1.0.0',
'installed' => true,
'enabled' => true,
]);
Registry::registerMenu('menu-module', [
'title' => 'menu_module::menu.title',
'link' => '/admin/modules/menu-module/settings',
'icon' => 'CalculatorIcon',
]);
$response = getJson('api/v1/company-modules')->assertOk();
$response->assertJsonPath('data.0.menu.title', 'menu_module::menu.title');
$response->assertJsonPath('data.0.menu.icon', 'CalculatorIcon');
});
test('returns empty array when no modules are enabled', function () {
getJson('api/v1/company-modules')
->assertOk()
->assertJsonPath('data', []);
});