1
0
Fork 0
mirror of https://github.com/archtechx/tenancy.git synced 2025-12-12 08:04:03 +00:00

Misc test fixes (#44)

* Add suffix_storage_path test

* Get filesystem bootstrapper coverage to 100%

* Delete enabling DB bootstrapper in TestCase

* Complete most of test todos

* Complete last tests todo

* Fix docblock

* add todo

---------

Co-authored-by: lukinovec <lukinovec@gmail.com>
This commit is contained in:
Samuel Štancl 2024-04-06 19:17:34 +02:00 committed by GitHub
parent 489fbb9402
commit d9ca3cec38
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
7 changed files with 154 additions and 159 deletions

View file

@ -15,7 +15,7 @@
<env name="APP_ENV" value="testing"/> <env name="APP_ENV" value="testing"/>
<env name="APP_KEY" value="base64:uYVmTs9lrQbXWfHgSSiG0VZMjc2KG/fBbjV1i1JDVos="/> <env name="APP_KEY" value="base64:uYVmTs9lrQbXWfHgSSiG0VZMjc2KG/fBbjV1i1JDVos="/>
<env name="BCRYPT_ROUNDS" value="4"/> <env name="BCRYPT_ROUNDS" value="4"/>
<env name="CACHE_DRIVER" value="redis"/> <env name="CACHE_STORE" value="redis"/>
<env name="MAIL_DRIVER" value="array"/> <env name="MAIL_DRIVER" value="array"/>
<env name="QUEUE_CONNECTION" value="sync"/> <env name="QUEUE_CONNECTION" value="sync"/>
<env name="SESSION_DRIVER" value="array"/> <env name="SESSION_DRIVER" value="array"/>

View file

@ -34,7 +34,7 @@ class FortifyRouteBootstrapper implements TenancyBootstrapper
* *
* For example: * For example:
* *
* FortifyRouteTenancyBootstrapper::$fortifyRedirectMap = [ * FortifyRouteBootstrapper::$fortifyRedirectMap = [
* // On logout, redirect the user to the "bye" route in the central app * // On logout, redirect the user to the "bye" route in the central app
* 'logout' => [ * 'logout' => [
* 'route_name' => 'bye', * 'route_name' => 'bye',
@ -46,7 +46,7 @@ class FortifyRouteBootstrapper implements TenancyBootstrapper
* 'route_name' => 'welcome', * 'route_name' => 'welcome',
* 'context' => Context::TENANT, * 'context' => Context::TENANT,
* ], * ],
*]; * ];
*/ */
public static array $fortifyRedirectMap = []; public static array $fortifyRedirectMap = [];

View file

@ -15,12 +15,18 @@ beforeEach(function () {
Event::listen(TenancyEnded::class, RevertToCentralContext::class); Event::listen(TenancyEnded::class, RevertToCentralContext::class);
}); });
test('context is switched when tenancy is initialized', function () { test('context is switched to tenant when initializing tenancy and reverted when ending tenancy', function () {
contextIsSwitchedWhenTenancyInitialized(); config(['tenancy.bootstrappers' => [
}); MyBootstrapper::class,
]]);
test('context is reverted when tenancy is ended', function () { $tenant = Tenant::create([
contextIsSwitchedWhenTenancyInitialized(); 'id' => 'acme',
]);
tenancy()->initialize($tenant);
expect(app('tenancy_initialized_for_tenant'))->toBe('acme');
tenancy()->end(); tenancy()->end();
@ -95,22 +101,6 @@ test('central helper doesnt change tenancy state when called in central context'
expect(tenant())->toBeNull(); expect(tenant())->toBeNull();
}); });
// todo@tests
function contextIsSwitchedWhenTenancyInitialized()
{
config(['tenancy.bootstrappers' => [
MyBootstrapper::class,
]]);
$tenant = Tenant::create([
'id' => 'acme',
]);
tenancy()->initialize($tenant);
expect(app('tenancy_initialized_for_tenant'))->toBe('acme');
}
class MyBootstrapper implements TenancyBootstrapper class MyBootstrapper implements TenancyBootstrapper
{ {
public function bootstrap(\Stancl\Tenancy\Contracts\Tenant $tenant): void public function bootstrap(\Stancl\Tenancy\Contracts\Tenant $tenant): void

View file

@ -319,6 +319,41 @@ test('files can get fetched using the storage url', function() {
expect(file_get_contents(public_path($url)))->toBe($centralFileContent); expect(file_get_contents(public_path($url)))->toBe($centralFileContent);
}); });
test('storage_path helper does not change if suffix_storage_path is off', function() {
$originalStoragePath = storage_path();
// todo@tests https://github.com/tenancy-for-laravel/v4/pull/44#issue-2228530362
config([
'tenancy.bootstrappers' => [FilesystemTenancyBootstrapper::class],
'tenancy.filesystem.suffix_storage_path' => false,
]);
tenancy()->initialize(Tenant::create());
$this->assertEquals($originalStoragePath, storage_path());
});
test('links to storage disks with a configured root are suffixed if not overridden', function() {
config([
'filesystems.disks.public.root' => 'http://sample-s3-url.com/my-app',
'tenancy.bootstrappers' => [
FilesystemTenancyBootstrapper::class,
],
'tenancy.filesystem.root_override.public' => null,
'tenancy.filesystem.url_override.public' => null,
]);
$tenant = Tenant::create();
$expectedStoragePath = storage_path() . '/tenant' . $tenant->getTenantKey(); // /tenant = suffix base
tenancy()->initialize($tenant);
// Check suffixing logic
expect(storage_path())->toEqual($expectedStoragePath);
});
test('create and delete storage symlinks jobs work', function() { test('create and delete storage symlinks jobs work', function() {
Event::listen( Event::listen(
TenantCreated::class, TenantCreated::class,

View file

@ -232,18 +232,43 @@ test('seed command works', function () {
}); });
}); });
test('database connection is switched to default', function () { test('database connection is switched to default after running commands', function (bool $initializeTenancy) {
databaseConnectionSwitchedToDefault(); $tenant = Tenant::create();
});
test('database connection is switched to default when tenancy has been initialized', function () { if ($initializeTenancy) {
tenancy()->initialize(Tenant::create()); tenancy()->initialize($tenant);
}
databaseConnectionSwitchedToDefault(); $originalDBName = DB::connection()->getDatabaseName();
});
Artisan::call('tenants:migrate');
expect(DB::connection()->getDatabaseName())->toBe($originalDBName);
Artisan::call('tenants:seed', ['--class' => ExampleSeeder::class]);
expect(DB::connection()->getDatabaseName())->toBe($originalDBName);
Artisan::call('tenants:rollback');
expect(DB::connection()->getDatabaseName())->toBe($originalDBName);
Artisan::call('tenants:migrate', ['--tenants' => [$tenant->getTenantKey()]]);
pest()->artisan("tenants:run --tenants={$tenant->getTenantKey()} 'foo foo --b=bar --c=xyz'");
expect(DB::connection()->getDatabaseName())->toBe($originalDBName);
})->with([
'tenancy initialized' => true,
'tenancy not initialized' => false,
]);
test('run command works', function () { test('run command works', function () {
runCommandWorks(); $id = Tenant::create()->getTenantKey();
Artisan::call('tenants:migrate', ['--tenants' => [$id]]);
pest()->artisan("tenants:run --tenants=$id 'foo foo --b=bar --c=xyz'")
->expectsOutput("User's name is Test user")
->expectsOutput('foo')
->expectsOutput('xyz');
}); });
test('install command works', function () { test('install command works', function () {
@ -404,35 +429,3 @@ test('migrate fresh command only deletes tenant databases if drop_tenant_databas
expect($tenantHasDatabase($tenant))->toBe($shouldHaveDBAfterMigrateFresh); expect($tenantHasDatabase($tenant))->toBe($shouldHaveDBAfterMigrateFresh);
} }
})->with([true, false]); })->with([true, false]);
// todo@tests
function runCommandWorks(): void
{
$id = Tenant::create()->getTenantKey();
Artisan::call('tenants:migrate', ['--tenants' => [$id]]);
pest()->artisan("tenants:run --tenants=$id 'foo foo --b=bar --c=xyz' ")
->expectsOutput("User's name is Test user")
->expectsOutput('foo')
->expectsOutput('xyz');
}
// todo@tests
function databaseConnectionSwitchedToDefault()
{
$originalDBName = DB::connection()->getDatabaseName();
Artisan::call('tenants:migrate');
expect(DB::connection()->getDatabaseName())->toBe($originalDBName);
Artisan::call('tenants:seed', ['--class' => ExampleSeeder::class]);
expect(DB::connection()->getDatabaseName())->toBe($originalDBName);
Artisan::call('tenants:rollback');
expect(DB::connection()->getDatabaseName())->toBe($originalDBName);
runCommandWorks();
expect(DB::connection()->getDatabaseName())->toBe($originalDBName);
}

View file

@ -35,30 +35,55 @@ beforeEach(function () {
}); });
test('primary models are scoped to the current tenant', function () { test('primary models are scoped to the current tenant', function () {
primaryModelsScopedToCurrentTenant(); // acme context
}); tenancy()->initialize($acme = Tenant::create([
'id' => 'acme',
]));
test('primary models are not scoped in the central context', function () { $post = Post::create(['text' => 'Foo']);
primaryModelsScopedToCurrentTenant();
expect($post->tenant_id)->toBe('acme');
expect($post->tenant->id)->toBe('acme');
$post = Post::first();
expect($post->tenant_id)->toBe('acme');
expect($post->tenant->id)->toBe('acme');
// ======================================
// foobar context
tenancy()->initialize(Tenant::create([
'id' => 'foobar',
]));
$post = Post::create(['text' => 'Bar']);
expect($post->tenant_id)->toBe('foobar');
expect($post->tenant->id)->toBe('foobar');
$post = Post::first();
expect($post->tenant_id)->toBe('foobar');
expect($post->tenant->id)->toBe('foobar');
// ======================================
// acme context again
tenancy()->initialize($acme);
$post = Post::first();
expect($post->tenant_id)->toBe('acme');
expect($post->tenant->id)->toBe('acme');
// Assert foobar models are inaccessible in acme context
expect(Post::count())->toBe(1);
// Primary models are not scoped in the central context
tenancy()->end(); tenancy()->end();
expect(Post::count())->toBe(2); expect(Post::count())->toBe(2);
}); });
test('secondary models are scoped to the current tenant when accessed via primary model', function () {
secondaryModelsAreScopedToCurrentTenant();
});
test('secondary models are not scoped to the current tenant when accessed directly', function () {
secondaryModelsAreScopedToCurrentTenant();
// We're in acme context
expect(tenant('id'))->toBe('acme');
expect(Comment::count())->toBe(2);
});
test('secondary models ARE scoped to the current tenant when accessed directly and parent relationship trait is used', function () { test('secondary models ARE scoped to the current tenant when accessed directly and parent relationship trait is used', function () {
$acme = Tenant::create([ $acme = Tenant::create([
'id' => 'acme', 'id' => 'acme',
@ -91,9 +116,37 @@ test('secondary models ARE scoped to the current tenant when accessed directly a
expect(ScopedComment::count())->toBe(2); expect(ScopedComment::count())->toBe(2);
}); });
test('secondary models are not scoped in the central context', function () { test('secondary models are scoped correctly', function () {
secondaryModelsAreScopedToCurrentTenant(); // Secondary models are scoped to the current tenant when accessed via primary model
// acme context
tenancy()->initialize($acme = Tenant::create([
'id' => 'acme',
]));
$post = Post::create(['text' => 'Foo']);
$post->comments()->create(['text' => 'Comment text']);
// ================
// foobar context
tenancy()->initialize(Tenant::create([
'id' => 'foobar',
]));
$post = Post::create(['text' => 'Bar']);
$post->comments()->create(['text' => 'Comment text 2']);
// ================
// acme context again
tenancy()->initialize($acme);
expect(Post::count())->toBe(1);
expect(Post::first()->comments->count())->toBe(1);
// Secondary models are not scoped to the current tenant when accessed directly
expect(tenant('id'))->toBe('acme');
expect(Comment::count())->toBe(2);
// secondary models are not scoped in the central context
tenancy()->end(); tenancy()->end();
expect(Comment::count())->toBe(2); expect(Comment::count())->toBe(2);
@ -225,80 +278,6 @@ test('the model returned by the tenant helper has unique and exists validation r
expect($existsFails)->toBeFalse(); expect($existsFails)->toBeFalse();
}); });
// todo@tests
function primaryModelsScopedToCurrentTenant()
{
// acme context
tenancy()->initialize($acme = Tenant::create([
'id' => 'acme',
]));
$post = Post::create(['text' => 'Foo']);
expect($post->tenant_id)->toBe('acme');
expect($post->tenant->id)->toBe('acme');
$post = Post::first();
expect($post->tenant_id)->toBe('acme');
expect($post->tenant->id)->toBe('acme');
// ======================================
// foobar context
tenancy()->initialize($foobar = Tenant::create([
'id' => 'foobar',
]));
$post = Post::create(['text' => 'Bar']);
expect($post->tenant_id)->toBe('foobar');
expect($post->tenant->id)->toBe('foobar');
$post = Post::first();
expect($post->tenant_id)->toBe('foobar');
expect($post->tenant->id)->toBe('foobar');
// ======================================
// acme context again
tenancy()->initialize($acme);
$post = Post::first();
expect($post->tenant_id)->toBe('acme');
expect($post->tenant->id)->toBe('acme');
// Assert foobar models are inaccessible in acme context
expect(Post::count())->toBe(1);
}
// todo@tests
function secondaryModelsAreScopedToCurrentTenant()
{
// acme context
tenancy()->initialize($acme = Tenant::create([
'id' => 'acme',
]));
$post = Post::create(['text' => 'Foo']);
$post->comments()->create(['text' => 'Comment text']);
// ================
// foobar context
tenancy()->initialize($foobar = Tenant::create([
'id' => 'foobar',
]));
$post = Post::create(['text' => 'Bar']);
$post->comments()->create(['text' => 'Comment text 2']);
// ================
// acme context again
tenancy()->initialize($acme);
expect(Post::count())->toBe(1);
expect(Post::first()->comments->count())->toBe(1);
}
class Tenant extends TestTenant class Tenant extends TestTenant
{ {
use HasScopedValidationRules; use HasScopedValidationRules;

View file

@ -18,9 +18,7 @@ use Stancl\Tenancy\Bootstrappers\RootUrlBootstrapper;
use Stancl\Tenancy\Bootstrappers\MailConfigBootstrapper; use Stancl\Tenancy\Bootstrappers\MailConfigBootstrapper;
use Stancl\Tenancy\Bootstrappers\RedisTenancyBootstrapper; use Stancl\Tenancy\Bootstrappers\RedisTenancyBootstrapper;
use Stancl\Tenancy\Bootstrappers\UrlGeneratorBootstrapper; use Stancl\Tenancy\Bootstrappers\UrlGeneratorBootstrapper;
use Stancl\Tenancy\Bootstrappers\DatabaseTenancyBootstrapper;
use Stancl\Tenancy\Bootstrappers\BroadcastingConfigBootstrapper; use Stancl\Tenancy\Bootstrappers\BroadcastingConfigBootstrapper;
use Stancl\Tenancy\Bootstrappers\FilesystemTenancyBootstrapper;
use Stancl\Tenancy\Bootstrappers\CacheTenancyBootstrapper; use Stancl\Tenancy\Bootstrappers\CacheTenancyBootstrapper;
abstract class TestCase extends \Orchestra\Testbench\TestCase abstract class TestCase extends \Orchestra\Testbench\TestCase