- Restrictions for team folders

This commit is contained in:
Čarodej
2022-01-05 17:52:08 +01:00
parent b4887cea0e
commit ec29764c3f
16 changed files with 275 additions and 137 deletions
+4 -2
View File
@@ -70,7 +70,7 @@
"/chunks/profile~chunks/settings-password.js": "/chunks/profile~chunks/settings-password.js?id=dfa4128d68360d5e1b3b", "/chunks/profile~chunks/settings-password.js": "/chunks/profile~chunks/settings-password.js?id=dfa4128d68360d5e1b3b",
"/chunks/purchase-code.js": "/chunks/purchase-code.js?id=9e948882ae2315eb6132", "/chunks/purchase-code.js": "/chunks/purchase-code.js?id=9e948882ae2315eb6132",
"/chunks/recent-uploads.js": "/chunks/recent-uploads.js?id=8577d4c771602671b38a", "/chunks/recent-uploads.js": "/chunks/recent-uploads.js?id=8577d4c771602671b38a",
"/chunks/settings.js": "/chunks/settings.js?id=4466d97a0211f77d9317", "/chunks/settings.js": "/chunks/settings.js?id=b8e396ca81993a2ac3c9",
"/chunks/settings-password.js": "/chunks/settings-password.js?id=b6eb94764cc7b47f835e", "/chunks/settings-password.js": "/chunks/settings-password.js?id=b6eb94764cc7b47f835e",
"/chunks/settings-storage.js": "/chunks/settings-storage.js?id=76b45c336e8e12b23e81", "/chunks/settings-storage.js": "/chunks/settings-storage.js?id=76b45c336e8e12b23e81",
"/chunks/settings~chunks/settings-password.js": "/chunks/settings~chunks/settings-password.js?id=aafc9cd6aa47b01bc25a", "/chunks/settings~chunks/settings-password.js": "/chunks/settings~chunks/settings-password.js?id=aafc9cd6aa47b01bc25a",
@@ -198,5 +198,7 @@
"/js/main.51906e2db8a6c36f5304.hot-update.js": "/js/main.51906e2db8a6c36f5304.hot-update.js", "/js/main.51906e2db8a6c36f5304.hot-update.js": "/js/main.51906e2db8a6c36f5304.hot-update.js",
"/js/main.c09b0f39fe6d0b43515b.hot-update.js": "/js/main.c09b0f39fe6d0b43515b.hot-update.js", "/js/main.c09b0f39fe6d0b43515b.hot-update.js": "/js/main.c09b0f39fe6d0b43515b.hot-update.js",
"/js/main.de28ca938da3825e62b3.hot-update.js": "/js/main.de28ca938da3825e62b3.hot-update.js", "/js/main.de28ca938da3825e62b3.hot-update.js": "/js/main.de28ca938da3825e62b3.hot-update.js",
"/js/main.be91e0ed3d68ecb1af06.hot-update.js": "/js/main.be91e0ed3d68ecb1af06.hot-update.js" "/js/main.be91e0ed3d68ecb1af06.hot-update.js": "/js/main.be91e0ed3d68ecb1af06.hot-update.js",
"/chunks/settings.91678a5fa053e1ee00a7.hot-update.js": "/chunks/settings.91678a5fa053e1ee00a7.hot-update.js",
"/js/main.000dc4eab65b819bb1e4.hot-update.js": "/js/main.000dc4eab65b819bb1e4.hot-update.js"
} }
+5
View File
@@ -4,6 +4,11 @@ const ValidatorHelpers = {
install(Vue) { install(Vue) {
Vue.prototype.$cantInviteMember = function (email, invitations) { Vue.prototype.$cantInviteMember = function (email, invitations) {
if (store.getters.config.subscriptionType === 'metered') {
return false
}
// Get max team members limitations // Get max team members limitations
let limit = store.getters.user.data.meta.limitations.max_team_members let limit = store.getters.user.data.meta.limitations.max_team_members
-22
View File
@@ -42,31 +42,9 @@
<span class="email">{{ user.data.attributes.email }}</span> <span class="email">{{ user.data.attributes.email }}</span>
</div> </div>
</div> </div>
<!--<div v-if="config.storageLimit && config.isSaaS && config.app_payments_active && !canShowIncompletePayment" class="headline-actions">
<ButtonBase @click.native="$openUpgradeOptions" class="upgrade-button" button-style="secondary" type="button">
{{ $t('global.upgrade_plan') }}
</ButtonBase>
</div>-->
</div> </div>
<!-- <ButtonBase @click.native="$openUpgradeOptions" class="upgrade-button" button-style="secondary" type="button">
{{ $t('global.upgrade_plan') }}
</ButtonBase>-->
<CardNavigation :pages="pages" class="-mx-1" /> <CardNavigation :pages="pages" class="-mx-1" />
<!--Incomplete Payment Warning-->
<!--<InfoBox v-if="canShowIncompletePayment" type="error" class="message-box">
<i18n path="incomplete_payment.description" tag="p">
<a :href="user.data.attributes.incomplete_payment">{{ $t('incomplete_payment.href') }}</a>
</i18n>
</InfoBox>-->
<!--Upgrade Storage Plan Warning-->
<!--<InfoBox v-if="canShowUpgradeWarning && !canShowIncompletePayment" type="error" class="message-box">
<p>{{ $t('upgrade_banner.title') }}</p>
</InfoBox>-->
</div> </div>
<!--Router Content--> <!--Router Content-->
@@ -3,6 +3,7 @@ namespace App\Limitations\Engines;
use App\Users\Models\User; use App\Users\Models\User;
use App\Limitations\LimitationEngine; use App\Limitations\LimitationEngine;
use Domain\Teams\Actions\CheckMaxTeamMembersLimitAction;
class DefaultLimitationEngine implements LimitationEngine class DefaultLimitationEngine implements LimitationEngine
{ {
@@ -20,8 +21,7 @@ class DefaultLimitationEngine implements LimitationEngine
); );
// 2. Check if storage usage exceed predefined capacity // 2. Check if storage usage exceed predefined capacity
return ! ($usedPercentage >= 100) return ! ($usedPercentage >= 100);
;
} }
public function canDownload(User $user): bool public function canDownload(User $user): bool
@@ -33,4 +33,14 @@ class DefaultLimitationEngine implements LimitationEngine
{ {
return true; return true;
} }
public function canCreateTeamFolder(User $user): bool
{
return true;
}
public function canInviteTeamMembers(User $user, array $newInvites): bool
{
return resolve(CheckMaxTeamMembersLimitAction::class)($user, $newInvites);
}
} }
@@ -3,6 +3,7 @@ namespace App\Limitations\Engines;
use App\Users\Models\User; use App\Users\Models\User;
use App\Limitations\LimitationEngine; use App\Limitations\LimitationEngine;
use Domain\Teams\Actions\CheckMaxTeamMembersLimitAction;
class FixedBillingLimitationEngine implements LimitationEngine class FixedBillingLimitationEngine implements LimitationEngine
{ {
@@ -27,4 +28,14 @@ class FixedBillingLimitationEngine implements LimitationEngine
{ {
return true; return true;
} }
public function canCreateTeamFolder(User $user): bool
{
return true;
}
public function canInviteTeamMembers(User $user, array $newInvites): bool
{
return resolve(CheckMaxTeamMembersLimitAction::class)($user, $newInvites);
}
} }
@@ -23,4 +23,15 @@ class MeteredBillingLimitationEngine implements LimitationEngine
// Disable create folder when user has more than 3 failed payments // Disable create folder when user has more than 3 failed payments
return ! ($user->failedPayments()->count() >= 3); return ! ($user->failedPayments()->count() >= 3);
} }
public function canCreateTeamFolder(User $user): bool
{
// Disable create folder when user has more than 3 failed payments
return ! ($user->failedPayments()->count() >= 3);
}
public function canInviteTeamMembers(User $user, array $newInvites): bool
{
return true;
}
} }
+4
View File
@@ -10,4 +10,8 @@ interface LimitationEngine
public function canDownload(User $user): bool; public function canDownload(User $user): bool;
public function canCreateFolder(User $user): bool; public function canCreateFolder(User $user): bool;
public function canCreateTeamFolder(User $user): bool;
public function canInviteTeamMembers(User $user, array $newInvites): bool;
} }
@@ -1,7 +1,6 @@
<?php <?php
namespace Domain\Files\Actions; namespace Domain\Files\Actions;
use App\Users\Exceptions\InvalidUserActionException;
use Illuminate\Support\Str; use Illuminate\Support\Str;
use Domain\Sharing\Models\Share; use Domain\Sharing\Models\Share;
use Illuminate\Support\Facades\Auth; use Illuminate\Support\Facades\Auth;
@@ -10,6 +9,7 @@ use Illuminate\Support\Facades\Storage;
use Domain\Files\Requests\UploadRequest; use Domain\Files\Requests\UploadRequest;
use Domain\Files\Models\File as UserFile; use Domain\Files\Models\File as UserFile;
use Domain\Traffic\Actions\RecordUploadAction; use Domain\Traffic\Actions\RecordUploadAction;
use App\Users\Exceptions\InvalidUserActionException;
class UploadFileAction class UploadFileAction
{ {
@@ -1,7 +1,6 @@
<?php <?php
namespace Domain\Files\Controllers; namespace Domain\Files\Controllers;
use App\Users\Exceptions\InvalidUserActionException;
use Domain\Files\Models\File; use Domain\Files\Models\File;
use Illuminate\Http\Response; use Illuminate\Http\Response;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
@@ -9,6 +8,7 @@ use Domain\Files\Requests\UploadRequest;
use Domain\Files\Resources\FileResource; use Domain\Files\Resources\FileResource;
use Domain\Files\Actions\UploadFileAction; use Domain\Files\Actions\UploadFileAction;
use Support\Demo\Actions\FakeUploadFileAction; use Support\Demo\Actions\FakeUploadFileAction;
use App\Users\Exceptions\InvalidUserActionException;
class UploadFileController extends Controller class UploadFileController extends Controller
{ {
@@ -29,6 +29,7 @@ class UploadFileController extends Controller
} }
try { try {
// Upload and store file record
$file = ($this->uploadFiles)($request); $file = ($this->uploadFiles)($request);
return response(new FileResource($file), 201); return response(new FileResource($file), 201);
@@ -5,7 +5,7 @@ use App\Users\Models\User;
class CheckMaxTeamMembersLimitAction class CheckMaxTeamMembersLimitAction
{ {
public function __invoke(array $invitations, User $user) public function __invoke(User $user, array $newInvites): bool
{ {
// Get user limitation summary // Get user limitation summary
$limits = $user->limitations->summary(); $limits = $user->limitations->summary();
@@ -14,7 +14,7 @@ class CheckMaxTeamMembersLimitAction
$allowedEmails = $limits['max_team_members']['meta']['allowed_emails']; $allowedEmails = $limits['max_team_members']['meta']['allowed_emails'];
// Get new email invites from request // Get new email invites from request
$invitationEmails = collect($invitations) $invitationEmails = collect($newInvites)
->pluck('email'); ->pluck('email');
// Count total unique members // Count total unique members
@@ -24,8 +24,6 @@ class CheckMaxTeamMembersLimitAction
->count(); ->count();
// Check if there is more unique members than total max team members are allowed // Check if there is more unique members than total max team members are allowed
if ($totalMembers > $limits['max_team_members']['total']) { return ! ($totalMembers > $limits['max_team_members']['total']);
abort(423, 'You exceed your members limit.');
}
} }
} }
@@ -7,7 +7,6 @@ use App\Http\Controllers\Controller;
use Domain\Teams\Models\TeamFolderMember; use Domain\Teams\Models\TeamFolderMember;
use Illuminate\Contracts\Routing\ResponseFactory; use Illuminate\Contracts\Routing\ResponseFactory;
use Domain\Teams\Requests\ConvertIntoTeamFolderRequest; use Domain\Teams\Requests\ConvertIntoTeamFolderRequest;
use Domain\Teams\Actions\CheckMaxTeamMembersLimitAction;
use Domain\Teams\Actions\InviteMembersIntoTeamFolderAction; use Domain\Teams\Actions\InviteMembersIntoTeamFolderAction;
use Domain\Teams\Actions\SetTeamFolderPropertyForAllChildrenAction; use Domain\Teams\Actions\SetTeamFolderPropertyForAllChildrenAction;
@@ -15,7 +14,6 @@ class ConvertFolderIntoTeamFolderController extends Controller
{ {
public function __construct( public function __construct(
public InviteMembersIntoTeamFolderAction $inviteMembers, public InviteMembersIntoTeamFolderAction $inviteMembers,
public CheckMaxTeamMembersLimitAction $checkMaxTeamMembersLimit,
public SetTeamFolderPropertyForAllChildrenAction $setTeamFolderPropertyForAllChildren, public SetTeamFolderPropertyForAllChildrenAction $setTeamFolderPropertyForAllChildren,
) { ) {
} }
@@ -25,7 +23,12 @@ class ConvertFolderIntoTeamFolderController extends Controller
Folder $folder Folder $folder
): ResponseFactory|Response { ): ResponseFactory|Response {
// Check if user didn't exceed max team members limit // Check if user didn't exceed max team members limit
($this->checkMaxTeamMembersLimit)($request->input('invitations'), $folder->owner); if (! $folder->owner->canInviteTeamMembers($request->input('invitations'))) {
return response([
'type' => 'error',
'message' => 'You exceed your members limit.',
], 401);
}
// Update root team folder // Update root team folder
$folder->update([ $folder->update([
@@ -17,7 +17,6 @@ use Domain\Folders\Resources\FolderCollection;
use Domain\Teams\Actions\UpdateInvitationsAction; use Domain\Teams\Actions\UpdateInvitationsAction;
use Illuminate\Contracts\Routing\ResponseFactory; use Illuminate\Contracts\Routing\ResponseFactory;
use Domain\Teams\Requests\CreateTeamFolderRequest; use Domain\Teams\Requests\CreateTeamFolderRequest;
use Domain\Teams\Actions\CheckMaxTeamMembersLimitAction;
use Domain\Teams\Requests\UpdateTeamFolderMembersRequest; use Domain\Teams\Requests\UpdateTeamFolderMembersRequest;
use Domain\Teams\Actions\InviteMembersIntoTeamFolderAction; use Domain\Teams\Actions\InviteMembersIntoTeamFolderAction;
use Domain\Teams\Actions\SetTeamFolderPropertyForAllChildrenAction; use Domain\Teams\Actions\SetTeamFolderPropertyForAllChildrenAction;
@@ -27,7 +26,6 @@ class TeamFoldersController extends Controller
public function __construct( public function __construct(
public InviteMembersIntoTeamFolderAction $inviteMembers, public InviteMembersIntoTeamFolderAction $inviteMembers,
public SetTeamFolderPropertyForAllChildrenAction $setTeamFolderPropertyForAllChildren, public SetTeamFolderPropertyForAllChildrenAction $setTeamFolderPropertyForAllChildren,
public CheckMaxTeamMembersLimitAction $checkMaxTeamMembersLimit,
) { ) {
} }
@@ -61,8 +59,21 @@ class TeamFoldersController extends Controller
): ResponseFactory | Response { ): ResponseFactory | Response {
$data = CreateTeamFolderData::fromRequest($request); $data = CreateTeamFolderData::fromRequest($request);
// Check if user can create team folder
if (! $request->user()->canCreateTeamFolder()) {
return response([
'type' => 'error',
'message' => 'This user action is not allowed.',
], 401);
}
// Check if user didn't exceed max team members limit // Check if user didn't exceed max team members limit
($this->checkMaxTeamMembersLimit)($data->invitations, $request->user()); if (! $request->user()->canInviteTeamMembers($data->invitations)) {
return response([
'type' => 'error',
'message' => 'You exceed your members limit.',
], 401);
}
// Create folder // Create folder
$folder = Folder::create([ $folder = Folder::create([
@@ -93,7 +104,12 @@ class TeamFoldersController extends Controller
$this->authorize('owner', $folder); $this->authorize('owner', $folder);
// Check if user didn't exceed max team members limit // Check if user didn't exceed max team members limit
($this->checkMaxTeamMembersLimit)($request->input('invitations'), $request->user()); if (! $request->user()->canInviteTeamMembers($request->input('invitations'))) {
return response([
'type' => 'error',
'message' => 'You exceed your members limit.',
], 401);
}
$updateInvitations( $updateInvitations(
$folder, $folder,
@@ -5,6 +5,7 @@ use Tests\TestCase;
use App\Users\Models\User; use App\Users\Models\User;
use Domain\Files\Models\File; use Domain\Files\Models\File;
use Domain\Settings\Models\Setting; use Domain\Settings\Models\Setting;
use Domain\Teams\Models\TeamFolderMember;
class DefaultLimitationTest extends TestCase class DefaultLimitationTest extends TestCase
{ {
@@ -105,4 +106,82 @@ class DefaultLimitationTest extends TestCase
'name' => 'New Folder', 'name' => 'New Folder',
]); ]);
} }
/**
* @test
*/
public function it_cant_invite_team_members_into_team_folder_because_user_exceeded_members_limit()
{
$user = User::factory()
->hasFolders([
'team_folder' => true,
])
->create();
TeamFolderMember::create([
'parent_id' => $user->folders[0]->id,
'user_id' => $user->id,
'permission' => 'owner',
]);
// Create team folder members
$members = User::factory()
->count(5)
->create()
->each(
fn ($member) => TeamFolderMember::factory()
->create([
'parent_id' => $user->folders[0]->id,
'user_id' => $member->id,
])
);
// Try invite new members, it has to fail
$this
->actingAs($user)
->post('/api/teams/folders', [
'name' => 'Company Project',
'invitations' => [
[
'email' => 'test@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test2@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test3@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test4@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test5@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test6@doe.com',
'permission' => 'can-edit',
],
],
])
->assertStatus(401);
// Invite existing member, it has to go through
$this
->actingAs($user)
->post('/api/teams/folders', [
'name' => 'Company Project',
'invitations' => [
[
'email' => $members[0]->email,
'permission' => 'can-edit',
],
],
])
->assertCreated();
}
} }
@@ -5,6 +5,7 @@ use Tests\TestCase;
use App\Users\Models\User; use App\Users\Models\User;
use Domain\Files\Models\File; use Domain\Files\Models\File;
use Domain\Settings\Models\Setting; use Domain\Settings\Models\Setting;
use Domain\Teams\Models\TeamFolderMember;
class FixedBillingLimitationTest extends TestCase class FixedBillingLimitationTest extends TestCase
{ {
@@ -13,7 +14,7 @@ class FixedBillingLimitationTest extends TestCase
parent::setUp(); parent::setUp();
Setting::updateOrCreate([ Setting::updateOrCreate([
'name' => 'subscription_type', 'name' => 'subscription_type',
], [ ], [
'value' => 'fixed', 'value' => 'fixed',
]); ]);
@@ -58,7 +59,7 @@ class FixedBillingLimitationTest extends TestCase
$this $this
->actingAs($user) ->actingAs($user)
->postJson('/api/create-folder', [ ->postJson('/api/create-folder', [
'name' => 'New Folder', 'name' => 'New Folder',
]) ])
->assertStatus(201); ->assertStatus(201);
@@ -66,4 +67,82 @@ class FixedBillingLimitationTest extends TestCase
'name' => 'New Folder', 'name' => 'New Folder',
]); ]);
} }
/**
* @test
*/
public function it_cant_invite_team_members_into_team_folder_because_user_exceeded_members_limit()
{
$user = User::factory()
->hasFolders([
'team_folder' => true,
])
->create();
TeamFolderMember::create([
'parent_id' => $user->folders[0]->id,
'user_id' => $user->id,
'permission' => 'owner',
]);
// Create team folder members
$members = User::factory()
->count(5)
->create()
->each(
fn ($member) => TeamFolderMember::factory()
->create([
'parent_id' => $user->folders[0]->id,
'user_id' => $member->id,
])
);
// Try invite new members, it has to fail
$this
->actingAs($user)
->post('/api/teams/folders', [
'name' => 'Company Project',
'invitations' => [
[
'email' => 'test@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test2@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test3@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test4@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test5@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test6@doe.com',
'permission' => 'can-edit',
],
],
])
->assertStatus(401);
// Invite existing member, it has to go through
$this
->actingAs($user)
->post('/api/teams/folders', [
'name' => 'Company Project',
'invitations' => [
[
'email' => $members[0]->email,
'permission' => 'can-edit',
],
],
])
->assertCreated();
}
} }
@@ -49,16 +49,30 @@ class MeteredBillingLimitationTest extends TestCase
$user = User::factory() $user = User::factory()
->create(); ->create();
// Create basic folder
$this $this
->actingAs($user) ->actingAs($user)
->postJson('/api/create-folder', [ ->postJson('/api/create-folder', [
'name' => 'New Folder', 'name' => 'New Folder',
]) ])
->assertStatus(201); ->assertStatus(201);
$this->assertDatabaseHas('folders', [ // Create team folder
'name' => 'New Folder', $this
]); ->actingAs($user)
->postJson('/api/teams/folders', [
'name' => 'New Team Folder',
'invitations' => [
[
'email' => 'john@doe.com',
'permission' => 'can-edit',
'type' => 'invitation',
],
],
])
->assertStatus(201);
$this->assertDatabaseCount('folders', 2);
} }
/** /**
@@ -70,15 +84,29 @@ class MeteredBillingLimitationTest extends TestCase
->hasFailedpayments(3) ->hasFailedpayments(3)
->create(); ->create();
// Create basic folder
$this $this
->actingAs($user) ->actingAs($user)
->postJson('/api/create-folder', [ ->postJson('/api/create-folder', [
'name' => 'New Folder', 'name' => 'New Folder',
]) ])
->assertStatus(401); ->assertStatus(401);
$this->assertDatabaseMissing('folders', [ // Create team folder
'name' => 'New Folder', $this
]); ->actingAs($user)
->postJson('/api/teams/folders', [
'name' => 'New Folder',
'invitations' => [
[
'email' => 'john@doe.com',
'permission' => 'can-edit',
'type' => 'invitation',
],
],
])
->assertStatus(401);
$this->assertDatabaseCount('folders', 0);
} }
} }
-87
View File
@@ -1,87 +0,0 @@
<?php
namespace Tests\Domain\Teams;
use Tests\TestCase;
use App\Users\Models\User;
use Domain\Teams\Models\TeamFolderMember;
class TeamLimitsTest extends TestCase
{
/**
* @test
*/
public function it_create_team_folder_with_exceeded_members_limit()
{
$user = User::factory()
->hasFolders([
'team_folder' => true,
])
->create();
TeamFolderMember::create([
'parent_id' => $user->folders[0]->id,
'user_id' => $user->id,
'permission' => 'owner',
]);
$members = User::factory()
->count(5)
->create();
$members->each(
fn ($member) => TeamFolderMember::factory()
->create([
'parent_id' => $user->folders[0]->id,
'user_id' => $member->id,
])
);
// Try invite new member
$this
->actingAs($user)
->post('/api/teams/folders', [
'name' => 'Company Project',
'invitations' => [
[
'email' => 'test@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test2@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test3@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test4@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test5@doe.com',
'permission' => 'can-edit',
],
[
'email' => 'test6@doe.com',
'permission' => 'can-edit',
],
],
])
->assertStatus(423);
// Invite existing member
$this
->actingAs($user)
->post('/api/teams/folders', [
'name' => 'Company Project',
'invitations' => [
[
'email' => $members[0]->email,
'permission' => 'can-edit',
],
],
])
->assertCreated();
}
}