mirror of
https://github.com/VueFileManager/vuefilemanager.git
synced 2026-04-05 18:23:48 +00:00
team members limitation frontend/backend
This commit is contained in:
@@ -26,7 +26,7 @@ class TeamFolderInvitationFactory extends Factory
|
||||
'parent_id' => $this->faker->uuid,
|
||||
'inviter_id' => $this->faker->uuid,
|
||||
'email' => $this->faker->email,
|
||||
'permission' => $this->faker->randomElement(['can-edit', 'can-view', 'can-view-and-download']),
|
||||
'permission' => $this->faker->randomElement(['can-edit', 'can-view']),
|
||||
'status' => $this->faker->randomElement(['pending', 'accepted', 'rejected']),
|
||||
];
|
||||
}
|
||||
|
||||
30
database/factories/TeamFolderMemberFactory.php
Normal file
30
database/factories/TeamFolderMemberFactory.php
Normal file
@@ -0,0 +1,30 @@
|
||||
<?php
|
||||
|
||||
namespace Database\Factories;
|
||||
|
||||
use Domain\Teams\Models\TeamFolderMember;
|
||||
use Illuminate\Database\Eloquent\Factories\Factory;
|
||||
|
||||
class TeamFolderMemberFactory extends Factory
|
||||
{
|
||||
/**
|
||||
* The name of the factory's corresponding model.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $model = TeamFolderMember::class;
|
||||
|
||||
/**
|
||||
* Define the model's default state.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function definition()
|
||||
{
|
||||
return [
|
||||
'parent_id' => $this->faker->uuid,
|
||||
'user_id' => $this->faker->uuid,
|
||||
'permission' => $this->faker->randomElement(['can-edit', 'can-view', 'owner']),
|
||||
];
|
||||
}
|
||||
}
|
||||
@@ -60,7 +60,7 @@
|
||||
"/chunks/plan-settings.js": "/chunks/plan-settings.js?id=c07a2a970dc5414a7d51",
|
||||
"/chunks/plan-subscribers.js": "/chunks/plan-subscribers.js?id=5151a942550b58d3ba69",
|
||||
"/chunks/plans.js": "/chunks/plans.js?id=0aa91553ed338315d39a",
|
||||
"/chunks/platform.js": "/chunks/platform.js?id=e25d3e8c863a76737a4b",
|
||||
"/chunks/platform.js": "/chunks/platform.js?id=ddb79fb6130d8c6d85fe",
|
||||
"/chunks/platform~chunks/shared.js": "/chunks/platform~chunks/shared.js?id=e64e3b67ccb89de9dd55",
|
||||
"/chunks/platform~chunks/shared~chunks/shared-with-me~chunks/team-folders.js": "/chunks/platform~chunks/shared~chunks/shared-with-me~chunks/team-folders.js?id=7d983dfdc91de607d737",
|
||||
"/chunks/profile.js": "/chunks/profile.js?id=8688d9c7ff850e6989e6",
|
||||
@@ -559,5 +559,12 @@
|
||||
"/js/main.c85b0e6c1c407b517efa.hot-update.js": "/js/main.c85b0e6c1c407b517efa.hot-update.js",
|
||||
"/js/main.6d054575b90412bd027d.hot-update.js": "/js/main.6d054575b90412bd027d.hot-update.js",
|
||||
"/js/main.1b9dd6c7b70765387854.hot-update.js": "/js/main.1b9dd6c7b70765387854.hot-update.js",
|
||||
"/js/main.1be136e3d8e2c392d8aa.hot-update.js": "/js/main.1be136e3d8e2c392d8aa.hot-update.js"
|
||||
"/js/main.1be136e3d8e2c392d8aa.hot-update.js": "/js/main.1be136e3d8e2c392d8aa.hot-update.js",
|
||||
"/chunks/platform.9b9c3e26dbe41d5f8465.hot-update.js": "/chunks/platform.9b9c3e26dbe41d5f8465.hot-update.js",
|
||||
"/chunks/platform.3b1ff5b0e9effcce5ee9.hot-update.js": "/chunks/platform.3b1ff5b0e9effcce5ee9.hot-update.js",
|
||||
"/chunks/platform.686e9ec6741d10aa9bc0.hot-update.js": "/chunks/platform.686e9ec6741d10aa9bc0.hot-update.js",
|
||||
"/chunks/platform.dfdf9807b5e5a42df629.hot-update.js": "/chunks/platform.dfdf9807b5e5a42df629.hot-update.js",
|
||||
"/chunks/platform.106a731410cf0cfa904f.hot-update.js": "/chunks/platform.106a731410cf0cfa904f.hot-update.js",
|
||||
"/chunks/platform.8317102f76e6a7ccb6dd.hot-update.js": "/chunks/platform.8317102f76e6a7ccb6dd.hot-update.js",
|
||||
"/chunks/platform.2ddb2dd6bf6a6992a1f4.hot-update.js": "/chunks/platform.2ddb2dd6bf6a6992a1f4.hot-update.js"
|
||||
}
|
||||
|
||||
@@ -91,6 +91,7 @@
|
||||
import InfoBox from "../Others/Forms/InfoBox";
|
||||
import {events} from '/resources/js/bus'
|
||||
import axios from "axios";
|
||||
import {mapGetters} from "vuex";
|
||||
|
||||
export default {
|
||||
name: 'CreateTeamFolderPopup',
|
||||
@@ -109,6 +110,9 @@
|
||||
InfoBox,
|
||||
},
|
||||
computed: {
|
||||
...mapGetters([
|
||||
'user'
|
||||
]),
|
||||
popupTitle() {
|
||||
return this.item ? this.$t('Convert as Team Folder') : this.$t('Create Team Folder')
|
||||
},
|
||||
@@ -198,6 +202,17 @@
|
||||
return
|
||||
}
|
||||
|
||||
// Get team folder limitations
|
||||
let limit = this.user.data.meta.limitations.max_team_members
|
||||
|
||||
if (limit.percentage >= 100 && ! limit.meta.allowed_emails.includes(email)) {
|
||||
this.$refs.teamFolderForm.setErrors({
|
||||
'Email': this.$t("You have to upgrade your account to add this new member.")
|
||||
});
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
this.$refs.teamFolderForm.reset()
|
||||
|
||||
this.invitations.push({
|
||||
|
||||
@@ -80,6 +80,7 @@
|
||||
import InfoBox from "../Others/Forms/InfoBox";
|
||||
import {events} from '/resources/js/bus'
|
||||
import axios from "axios";
|
||||
import {mapGetters} from "vuex";
|
||||
|
||||
export default {
|
||||
name: 'EditTeamFolderPopup',
|
||||
@@ -98,6 +99,9 @@
|
||||
InfoBox,
|
||||
},
|
||||
computed: {
|
||||
...mapGetters([
|
||||
'user'
|
||||
]),
|
||||
isDisabledSubmit() {
|
||||
return Object.values(this.members).length === 0 && Object.values(this.invitations).length === 0
|
||||
}
|
||||
@@ -157,6 +161,17 @@
|
||||
});
|
||||
}
|
||||
|
||||
// Get team folder limitations
|
||||
let limit = this.user.data.meta.limitations.max_team_members
|
||||
|
||||
if (limit.percentage >= 100 && ! limit.meta.allowed_emails.includes(email)) {
|
||||
this.$refs.teamFolderForm.setErrors({
|
||||
'Email': this.$t("You have to upgrade your account to add this new member.")
|
||||
});
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
this.$refs.teamFolderForm.reset()
|
||||
|
||||
this.invitations.push({
|
||||
|
||||
@@ -109,34 +109,6 @@ class User extends Authenticatable implements MustVerifyEmail
|
||||
];
|
||||
}
|
||||
|
||||
// TODO: caching & refactoring
|
||||
public function accountLimitations(): array
|
||||
{
|
||||
$members = \DB::table('team_folder_members')
|
||||
->where('user_id', $this->id)
|
||||
->pluck('parent_id');
|
||||
|
||||
$membersUse = \DB::table('team_folder_members')
|
||||
->where('user_id', '!=', $this->id)
|
||||
->whereIn('parent_id', $members)
|
||||
->pluck('user_id')
|
||||
->unique()
|
||||
->count();
|
||||
|
||||
return [
|
||||
'max_storage_amount' => [
|
||||
'use' => Metric::bytes($this->usedCapacity)->format(),
|
||||
'total' => format_gigabytes($this->limitations->max_storage_amount),
|
||||
'percentage' => (float)get_storage_fill_percentage($this->usedCapacity, $this->limitations->max_storage_amount),
|
||||
],
|
||||
'max_team_members' => [
|
||||
'use' => $membersUse,
|
||||
'total' => (int)$this->limitations->max_team_members,
|
||||
'percentage' => ($membersUse / $this->limitations->max_team_members) * 100,
|
||||
],
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Get user used storage capacity in bytes
|
||||
*/
|
||||
@@ -209,6 +181,11 @@ class User extends Authenticatable implements MustVerifyEmail
|
||||
return $this->hasMany(File::class);
|
||||
}
|
||||
|
||||
public function folders(): HasMany
|
||||
{
|
||||
return $this->hasMany(Folder::class);
|
||||
}
|
||||
|
||||
/**
|
||||
* Send the password reset notification.
|
||||
*/
|
||||
@@ -227,7 +204,7 @@ class User extends Authenticatable implements MustVerifyEmail
|
||||
// Create default limitations
|
||||
$user->limitations()->create([
|
||||
'max_storage_amount' => get_settings('default_storage_amount') ?? 1,
|
||||
'max_team_members' => 3,
|
||||
'max_team_members' => 5,
|
||||
]);
|
||||
|
||||
// Create user directory for his files
|
||||
|
||||
@@ -1,10 +1,19 @@
|
||||
<?php
|
||||
|
||||
namespace App\Users\Models;
|
||||
|
||||
use ByteUnits\Metric;
|
||||
use DB;
|
||||
use Illuminate\Database\Eloquent\Model;
|
||||
use Database\Factories\UserLimitationFactory;
|
||||
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
||||
use Illuminate\Database\Eloquent\Relations\HasOne;
|
||||
|
||||
/**
|
||||
* @property int max_storage_amount
|
||||
* @property int max_team_members
|
||||
* @property string user_id
|
||||
*/
|
||||
class UserLimitation extends Model
|
||||
{
|
||||
use HasFactory;
|
||||
@@ -14,7 +23,7 @@ class UserLimitation extends Model
|
||||
protected $guarded = [];
|
||||
|
||||
protected $hidden = [
|
||||
'user_id',
|
||||
'user_id', 'user'
|
||||
];
|
||||
|
||||
public $incrementing = false;
|
||||
@@ -25,4 +34,74 @@ class UserLimitation extends Model
|
||||
{
|
||||
return UserLimitationFactory::new();
|
||||
}
|
||||
|
||||
public function user(): HasOne
|
||||
{
|
||||
return $this->hasOne(User::class, 'id', 'user_id');
|
||||
}
|
||||
|
||||
/**
|
||||
* Get summary of user limitations and their usage
|
||||
*/
|
||||
public function summary(): array
|
||||
{
|
||||
return [
|
||||
'max_storage_amount' => $this->getMaxStorageAmount(),
|
||||
'max_team_members' => $this->getMaxTeamMembers(),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Get usage data of user storage
|
||||
*/
|
||||
private function getMaxStorageAmount(): array
|
||||
{
|
||||
$userCapacity = $this->user->usedCapacity;
|
||||
|
||||
return [
|
||||
'use' => Metric::bytes($userCapacity)->format(),
|
||||
'total' => format_gigabytes($this->max_storage_amount),
|
||||
'percentage' => get_storage_fill_percentage($userCapacity, $this->max_storage_amount),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Get usage data of team members
|
||||
*/
|
||||
private function getMaxTeamMembers(): array
|
||||
{
|
||||
$userTeamFolderIds = DB::table('team_folder_members')
|
||||
->where('user_id', $this->user_id)
|
||||
->pluck('parent_id');
|
||||
|
||||
$memberIds = DB::table('team_folder_members')
|
||||
->where('user_id', '!=', $this->user_id)
|
||||
->whereIn('parent_id', $userTeamFolderIds)
|
||||
->pluck('user_id')
|
||||
->unique();
|
||||
|
||||
// Get member emails
|
||||
$memberEmails = User::whereIn('id', $memberIds)
|
||||
->pluck('email');
|
||||
|
||||
// Get active invitation emails
|
||||
$InvitationEmails = DB::table('team_folder_invitations')
|
||||
->where('status', 'pending')
|
||||
->where('inviter_id', $this->user_id)
|
||||
->pluck('email')
|
||||
->unique();
|
||||
|
||||
// Get allowed emails in the limit
|
||||
$totalUsedEmails = $memberEmails->merge($InvitationEmails)
|
||||
->unique();
|
||||
|
||||
return [
|
||||
'use' => $totalUsedEmails->count(),
|
||||
'total' => (int) $this->max_team_members,
|
||||
'percentage' => ($totalUsedEmails->count() / $this->max_team_members) * 100,
|
||||
'meta' => [
|
||||
'allowed_emails' => $totalUsedEmails,
|
||||
],
|
||||
];
|
||||
}
|
||||
}
|
||||
|
||||
@@ -45,7 +45,7 @@ class UserResource extends JsonResource
|
||||
]),
|
||||
],
|
||||
'meta' => [
|
||||
'limitations' => $this->accountLimitations(),
|
||||
'limitations' => $this->limitations->summary(),
|
||||
],
|
||||
],
|
||||
];
|
||||
|
||||
32
src/Domain/Teams/Actions/CheckMaxTeamMembersLimitAction.php
Normal file
32
src/Domain/Teams/Actions/CheckMaxTeamMembersLimitAction.php
Normal file
@@ -0,0 +1,32 @@
|
||||
<?php
|
||||
|
||||
namespace Domain\Teams\Actions;
|
||||
|
||||
use App\Users\Models\User;
|
||||
|
||||
class CheckMaxTeamMembersLimitAction
|
||||
{
|
||||
public function __invoke(array $invitations, User $user)
|
||||
{
|
||||
// Get user limitation summary
|
||||
$limits = $user->limitations->summary();
|
||||
|
||||
// Get currently used member emails
|
||||
$allowedEmails = $limits['max_team_members']['meta']['allowed_emails'];
|
||||
|
||||
// Get new email invites from request
|
||||
$invitationEmails = collect($invitations)
|
||||
->pluck('email');
|
||||
|
||||
// Count total unique members
|
||||
$totalMembers = $allowedEmails
|
||||
->merge($invitationEmails)
|
||||
->unique()
|
||||
->count();
|
||||
|
||||
// Check if there is more unique members than total max team members are allowed
|
||||
if ($totalMembers > $limits['max_team_members']['total']) {
|
||||
abort(423, 'You exceed your members limit.');
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,9 +1,10 @@
|
||||
<?php
|
||||
namespace Domain\Teams\Controllers;
|
||||
|
||||
use Domain\Teams\Actions\CheckMaxTeamMembersLimitAction;
|
||||
use Domain\Teams\Models\TeamFolderMember;
|
||||
use Illuminate\Http\Response;
|
||||
use Domain\Folders\Models\Folder;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use App\Http\Controllers\Controller;
|
||||
use Illuminate\Contracts\Routing\ResponseFactory;
|
||||
use Domain\Teams\Requests\ConvertIntoTeamFolderRequest;
|
||||
@@ -14,6 +15,7 @@ class ConvertFolderIntoTeamFolderController extends Controller
|
||||
{
|
||||
public function __construct(
|
||||
public InviteMembersIntoTeamFolderAction $inviteMembers,
|
||||
public CheckMaxTeamMembersLimitAction $checkMaxTeamMembersLimit,
|
||||
public SetTeamFolderPropertyForAllChildrenAction $setTeamFolderPropertyForAllChildren,
|
||||
) {
|
||||
}
|
||||
@@ -22,21 +24,24 @@ class ConvertFolderIntoTeamFolderController extends Controller
|
||||
ConvertIntoTeamFolderRequest $request,
|
||||
Folder $folder
|
||||
): ResponseFactory|Response {
|
||||
// Check if user didn't exceed max team members limit
|
||||
($this->checkMaxTeamMembersLimit)($request->input('invitations'), $folder->owner);
|
||||
|
||||
// Update root team folder
|
||||
$folder->update([
|
||||
'team_folder' => 1,
|
||||
'parent_id' => null,
|
||||
]);
|
||||
|
||||
// Mark all children folders as team folder
|
||||
($this->setTeamFolderPropertyForAllChildren)($folder, true);
|
||||
|
||||
// Attach owner into members
|
||||
DB::table('team_folder_members')
|
||||
->insert([
|
||||
'parent_id' => $folder->id,
|
||||
'user_id' => $folder->user_id,
|
||||
'permission' => 'owner',
|
||||
]);
|
||||
TeamFolderMember::create([
|
||||
'parent_id' => $folder->id,
|
||||
'user_id' => $folder->user_id,
|
||||
'permission' => 'owner',
|
||||
]);
|
||||
|
||||
// Invite team members
|
||||
($this->inviteMembers)($request->input('invitations'), $folder);
|
||||
|
||||
@@ -1,6 +1,8 @@
|
||||
<?php
|
||||
namespace Domain\Teams\Controllers;
|
||||
|
||||
use Domain\Teams\Actions\CheckMaxTeamMembersLimitAction;
|
||||
use Domain\Teams\Models\TeamFolderMember;
|
||||
use Illuminate\Support\Str;
|
||||
use Domain\Files\Models\File;
|
||||
use Illuminate\Http\Response;
|
||||
@@ -25,6 +27,7 @@ class TeamFoldersController extends Controller
|
||||
public function __construct(
|
||||
public InviteMembersIntoTeamFolderAction $inviteMembers,
|
||||
public SetTeamFolderPropertyForAllChildrenAction $setTeamFolderPropertyForAllChildren,
|
||||
public CheckMaxTeamMembersLimitAction $checkMaxTeamMembersLimit,
|
||||
) {
|
||||
}
|
||||
|
||||
@@ -58,6 +61,10 @@ class TeamFoldersController extends Controller
|
||||
): ResponseFactory | Response {
|
||||
$data = CreateTeamFolderData::fromRequest($request);
|
||||
|
||||
// Check if user didn't exceed max team members limit
|
||||
($this->checkMaxTeamMembersLimit)($data->invitations, $request->user());
|
||||
|
||||
// Create folder
|
||||
$folder = Folder::create([
|
||||
'user_id' => $request->user()->id,
|
||||
'name' => $data->name,
|
||||
@@ -65,12 +72,11 @@ class TeamFoldersController extends Controller
|
||||
]);
|
||||
|
||||
// Attach owner into members
|
||||
DB::table('team_folder_members')
|
||||
->insert([
|
||||
'parent_id' => $folder->id,
|
||||
'user_id' => $request->user()->id,
|
||||
'permission' => 'owner',
|
||||
]);
|
||||
TeamFolderMember::create([
|
||||
'parent_id' => $folder->id,
|
||||
'user_id' => $request->user()->id,
|
||||
'permission' => 'owner',
|
||||
]);
|
||||
|
||||
// Invite team members
|
||||
$this->inviteMembers->onQueue()->execute($data->invitations, $folder);
|
||||
@@ -86,6 +92,9 @@ class TeamFoldersController extends Controller
|
||||
): ResponseFactory | Response {
|
||||
$this->authorize('owner', $folder);
|
||||
|
||||
// Check if user didn't exceed max team members limit
|
||||
($this->checkMaxTeamMembersLimit)($request->input('invitations'), $request->user());
|
||||
|
||||
$updateInvitations(
|
||||
$folder,
|
||||
$request->input('invitations')
|
||||
|
||||
31
src/Domain/Teams/Models/TeamFolderMember.php
Normal file
31
src/Domain/Teams/Models/TeamFolderMember.php
Normal file
@@ -0,0 +1,31 @@
|
||||
<?php
|
||||
namespace Domain\Teams\Models;
|
||||
|
||||
use Database\Factories\TeamFolderMemberFactory;
|
||||
use Illuminate\Database\Eloquent\Model;
|
||||
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
||||
|
||||
/**
|
||||
* @method static create(array $array)
|
||||
* @property string id
|
||||
* @property string parent_id
|
||||
* @property string email
|
||||
* @property string status
|
||||
* @property string created_at
|
||||
* @property string updated_at
|
||||
*/
|
||||
class TeamFolderMember extends Model
|
||||
{
|
||||
use HasFactory;
|
||||
|
||||
protected $guarded = [];
|
||||
|
||||
public $incrementing = false;
|
||||
|
||||
public $timestamps = false;
|
||||
|
||||
protected static function newFactory(): TeamFolderMemberFactory
|
||||
{
|
||||
return TeamFolderMemberFactory::new();
|
||||
}
|
||||
}
|
||||
@@ -477,7 +477,7 @@ if (! function_exists('get_storage_fill_percentage')) {
|
||||
}
|
||||
|
||||
// Return in 2 decimal
|
||||
return number_format((float) $progress, 2, '.', '');
|
||||
return (float) number_format((float) $progress, 2, '.', '');
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
<?php
|
||||
namespace Tests\Feature\Accounts;
|
||||
|
||||
namespace Tests\App\Users;
|
||||
|
||||
use Storage;
|
||||
use Notification;
|
||||
@@ -117,8 +118,7 @@ class UserAccountTest extends TestCase
|
||||
|
||||
collect(config('vuefilemanager.avatar_sizes'))
|
||||
->each(
|
||||
fn ($size) =>
|
||||
Storage::disk('local')
|
||||
fn($size) => Storage::disk('local')
|
||||
->assertExists("avatars/{$size['name']}-{$user->settings->getRawOriginal('avatar')}")
|
||||
);
|
||||
}
|
||||
@@ -137,26 +137,27 @@ class UserAccountTest extends TestCase
|
||||
->assertStatus(200)
|
||||
->assertExactJson([
|
||||
'data' => [
|
||||
'id' => (string) $user->id,
|
||||
'id' => (string)$user->id,
|
||||
'type' => 'user',
|
||||
'attributes' => [
|
||||
'email' => $user->email,
|
||||
'role' => $user->role,
|
||||
'two_factor_authentication' => false,
|
||||
'folders' => [],
|
||||
'storage' => [
|
||||
'avatar' => null,
|
||||
'email' => $user->email,
|
||||
'role' => $user->role,
|
||||
'two_factor_authentication' => false,
|
||||
'folders' => [],
|
||||
'storage' => [
|
||||
'used' => 0,
|
||||
'used_formatted' => '0.00%',
|
||||
'used_formatted' => '0%',
|
||||
'capacity' => '1',
|
||||
'capacity_formatted' => '1GB',
|
||||
],
|
||||
'created_at' => format_date($user->created_at, '%d. %B. %Y'),
|
||||
'updated_at' => format_date($user->updated_at, '%d. %B. %Y'),
|
||||
'created_at' => format_date($user->created_at, '%d. %b. %Y'),
|
||||
'updated_at' => format_date($user->updated_at, '%d. %b. %Y'),
|
||||
],
|
||||
'relationships' => [
|
||||
'settings' => [
|
||||
'settings' => [
|
||||
'data' => [
|
||||
'id' => (string) $user->id,
|
||||
'id' => (string)$user->id,
|
||||
'type' => 'settings',
|
||||
'attributes' => [
|
||||
'avatar' => $user->settings->avatar,
|
||||
@@ -171,13 +172,13 @@ class UserAccountTest extends TestCase
|
||||
],
|
||||
],
|
||||
],
|
||||
'favourites' => [
|
||||
'favourites' => [
|
||||
'data' => [],
|
||||
],
|
||||
'limitations' => [
|
||||
'id' => $user->id,
|
||||
'type' => 'limitations',
|
||||
'data' => [
|
||||
'id' => $user->id,
|
||||
'type' => 'limitations',
|
||||
'data' => [
|
||||
'attributes' => $user->limitations,
|
||||
],
|
||||
],
|
||||
|
||||
@@ -32,6 +32,7 @@ class DashboardTest extends TestCase
|
||||
->assertStatus(200)
|
||||
->assertExactJson([
|
||||
'license' => 'Regular',
|
||||
'total_premium_users' => 0,
|
||||
'app_version' => config('vuefilemanager.version'),
|
||||
'total_users' => 1,
|
||||
'total_used_space' => '2.00MB',
|
||||
|
||||
67
tests/Domain/Teams/TeamLimitsTest.php
Normal file
67
tests/Domain/Teams/TeamLimitsTest.php
Normal file
@@ -0,0 +1,67 @@
|
||||
<?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',
|
||||
],
|
||||
],
|
||||
])
|
||||
->assertStatus(423);
|
||||
|
||||
// Invite existing member
|
||||
$this
|
||||
->actingAs($user)
|
||||
->post('/api/teams/folders', [
|
||||
'name' => 'Company Project',
|
||||
'invitations' => [
|
||||
[
|
||||
'email' => $members[0]->email,
|
||||
'permission' => 'can-edit',
|
||||
],
|
||||
],
|
||||
])
|
||||
->assertCreated();
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user