mirror of
https://github.com/HolgerHatGarKeineNode/einundzwanzig-app.git
synced 2026-05-18 20:34:52 +00:00
🔄 **Refactor and extend meetup membership-based authorization**
- Updated `authorizeAccess` to restrict `meetups.edit` views and updates to users in "My-Meetups". - Attached creators to `meetup_user` pivot for default membership. - Adjusted related tests to validate membership-based edit permissions. 📱 **Improve sidebar and mobile navigation accessibility** - Added `aria-labels` to improve screen reader support for sidebar and mobile header elements. - Updated desktop and mobile user menus alignment for consistency. ⚡ **Enhance Lightning login flow** - Introduced `lightningLoginInProgress` for smoother polling synchronization with the redirect flow. - Updated logic to dispatch `lightning-login-ready` event instead of immediate redirect, avoiding race conditions.
This commit is contained in:
@@ -10,6 +10,11 @@ export default () => ({
|
|||||||
// session-id migration (which would otherwise yield a 419 on the next
|
// session-id migration (which would otherwise yield a 419 on the next
|
||||||
// round-trip).
|
// round-trip).
|
||||||
nostrLoginInProgress: false,
|
nostrLoginInProgress: false,
|
||||||
|
// Toggled by the @lightning-login-ready handler in login.blade.php once
|
||||||
|
// the server signals a matching LoginKey. Same purpose: pause wire:poll
|
||||||
|
// before the full-page navigation to /auth/complete-lightning so the
|
||||||
|
// browser does not race the redirect with another poll tick.
|
||||||
|
lightningLoginInProgress: false,
|
||||||
|
|
||||||
async init() {
|
async init() {
|
||||||
this.startTime = Date.now();
|
this.startTime = Date.now();
|
||||||
|
|||||||
@@ -7,11 +7,18 @@
|
|||||||
<flux:toast.group>
|
<flux:toast.group>
|
||||||
<flux:toast/>
|
<flux:toast/>
|
||||||
</flux:toast.group>
|
</flux:toast.group>
|
||||||
<flux:sidebar sticky stashable class="border-e border-zinc-200 bg-zinc-50 dark:border-zinc-700 dark:bg-zinc-900">
|
<flux:sidebar
|
||||||
<flux:sidebar.toggle class="lg:hidden" icon="x-mark"/>
|
sticky
|
||||||
|
stashable
|
||||||
|
aria-label="{{ __('Hauptnavigation') }}"
|
||||||
|
class="border-e border-zinc-200 bg-zinc-50 pb-[max(1rem,calc(env(safe-area-inset-bottom)+1rem))] dark:border-zinc-700 dark:bg-zinc-900"
|
||||||
|
>
|
||||||
|
<flux:sidebar.toggle class="lg:hidden" icon="x-mark" aria-label="{{ __('Menü schließen') }}"/>
|
||||||
|
|
||||||
<a href="{{ route('welcome') }}" class="me-5 flex items-center space-x-2 rtl:space-x-reverse"
|
<a href="{{ route('welcome') }}"
|
||||||
wire:navigate>
|
class="me-5 flex items-center space-x-2 rtl:space-x-reverse"
|
||||||
|
wire:navigate
|
||||||
|
aria-label="{{ __('Zur Startseite') }}">
|
||||||
<x-app-logo/>
|
<x-app-logo/>
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
@@ -148,14 +155,15 @@
|
|||||||
</flux:navlist.group>
|
</flux:navlist.group>
|
||||||
</flux:navlist>
|
</flux:navlist>
|
||||||
|
|
||||||
<!-- Desktop User Menu -->
|
<!-- User Menu (Desktop + Mobile in sidebar) -->
|
||||||
@auth
|
@auth
|
||||||
<flux:dropdown class="hidden lg:block" position="bottom" align="start">
|
<flux:dropdown position="top" align="start">
|
||||||
<flux:profile
|
<flux:profile
|
||||||
:name="auth()->user()->name"
|
:name="auth()->user()->name"
|
||||||
:avatar="auth()->user()->profile_photo_url"
|
:avatar="auth()->user()->profile_photo_url"
|
||||||
:initials="auth()->user()->initials()"
|
:initials="auth()->user()->initials()"
|
||||||
icon:trailing="chevrons-up-down"
|
icon:trailing="chevrons-up-down"
|
||||||
|
aria-label="{{ __('Benutzermenü') }}"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<flux:menu class="w-[220px]">
|
<flux:menu class="w-[220px]">
|
||||||
@@ -201,66 +209,18 @@
|
|||||||
@endauth
|
@endauth
|
||||||
</flux:sidebar>
|
</flux:sidebar>
|
||||||
|
|
||||||
<!-- Mobile User Menu -->
|
<!-- Mobile Header (Toggle + Quick Country Chooser) -->
|
||||||
@auth
|
@auth
|
||||||
<flux:header class="lg:hidden">
|
<flux:header class="lg:hidden">
|
||||||
<flux:sidebar.toggle class="lg:hidden" icon="bars-2" inset="left"/>
|
<flux:sidebar.toggle class="lg:hidden" icon="bars-2" inset="left" aria-label="{{ __('Menü öffnen') }}"/>
|
||||||
|
|
||||||
<flux:spacer/>
|
<flux:spacer/>
|
||||||
|
|
||||||
<flux:navlist variant="outline" class="mr-6">
|
<flux:navlist variant="outline" class="me-1">
|
||||||
<flux:navlist.group class="grid">
|
<flux:navlist.group class="grid">
|
||||||
<livewire:country.chooser/>
|
<livewire:country.chooser/>
|
||||||
</flux:navlist.group>
|
</flux:navlist.group>
|
||||||
</flux:navlist>
|
</flux:navlist>
|
||||||
|
|
||||||
<flux:dropdown position="top" align="end">
|
|
||||||
<flux:profile
|
|
||||||
:avatar="auth()->user()->profile_photo_url"
|
|
||||||
:initials="auth()->user()->initials()"
|
|
||||||
icon-trailing="chevron-down"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<flux:menu>
|
|
||||||
<flux:menu.radio.group>
|
|
||||||
<div class="p-0 text-sm font-normal">
|
|
||||||
<div class="flex items-center gap-2 px-1 py-1.5 text-start text-sm">
|
|
||||||
<flux:avatar :src="auth()->user()->profile_photo_url" size="sm" class="shrink-0"/>
|
|
||||||
|
|
||||||
<div class="grid flex-1 text-start text-sm leading-tight">
|
|
||||||
<span class="truncate font-semibold">{{ auth()->user()?->name }}</span>
|
|
||||||
<span class="truncate text-xs">
|
|
||||||
@if(strlen(auth()->user()?->name) > 12)
|
|
||||||
{{ Str::substr(auth()->user()?->name, 0, 4) }}
|
|
||||||
...{{ Str::substr(auth()->user()?->name, -4) }}
|
|
||||||
@else
|
|
||||||
{{ auth()->user()?->name }}
|
|
||||||
@endif
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</flux:menu.radio.group>
|
|
||||||
|
|
||||||
<flux:menu.separator/>
|
|
||||||
|
|
||||||
<flux:menu.radio.group>
|
|
||||||
<flux:menu.item
|
|
||||||
:href="route('settings.profile', ['country' => str(session('lang_country', 'de'))->after('-')->lower()])"
|
|
||||||
icon="cog"
|
|
||||||
wire:navigate>{{ __('Settings') }}</flux:menu.item>
|
|
||||||
</flux:menu.radio.group>
|
|
||||||
|
|
||||||
<flux:menu.separator/>
|
|
||||||
|
|
||||||
<form method="POST" action="{{ route('logout') }}" class="w-full">
|
|
||||||
@csrf
|
|
||||||
<flux:menu.item as="button" type="submit" icon="arrow-right-start-on-rectangle" class="w-full">
|
|
||||||
{{ __('Log Out') }}
|
|
||||||
</flux:menu.item>
|
|
||||||
</form>
|
|
||||||
</flux:menu>
|
|
||||||
</flux:dropdown>
|
|
||||||
</flux:header>
|
</flux:header>
|
||||||
@endauth
|
@endauth
|
||||||
|
|
||||||
|
|||||||
@@ -282,42 +282,35 @@ class extends Component {
|
|||||||
return Str::transliterate(Str::lower($this->email).'|'.request()->ip());
|
return Str::transliterate(Str::lower($this->email).'|'.request()->ip());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function checkAuth()
|
public function checkAuth(): void
|
||||||
{
|
{
|
||||||
$loginKey = LoginKey::query()
|
$loginKey = LoginKey::query()
|
||||||
->where('k1', $this->k1)
|
->where('k1', $this->k1)
|
||||||
->whereDate('created_at', '>=', now()->subMinutes(5))
|
->where('created_at', '>=', now()->subMinutes(5))
|
||||||
->first();
|
->first();
|
||||||
|
|
||||||
if ($loginKey) {
|
if (! $loginKey) {
|
||||||
// Persist the locale choice before the auth round-trip — once we
|
return;
|
||||||
// redirect, this component is unmounted and $currentLangCountry
|
|
||||||
// would otherwise be lost.
|
|
||||||
session(['lang_country' => $this->currentLangCountry]);
|
|
||||||
|
|
||||||
// Hand off to a dedicated controller via full-page redirect.
|
|
||||||
// Calling auth()->login() inside the wire:poll handler rotates
|
|
||||||
// the session id and CSRF token mid-flight. Any Livewire request
|
|
||||||
// that arrives in the same window — a parallel wire:poll tick,
|
|
||||||
// a sibling component update, a click on the Nostr button —
|
|
||||||
// would then 419 (TokenMismatch). The controller performs the
|
|
||||||
// login on a clean, non-Livewire request before redirecting on
|
|
||||||
// to the dashboard.
|
|
||||||
return $this->redirect(
|
|
||||||
route('auth.ln.complete', ['k1' => $this->k1]),
|
|
||||||
navigate: false,
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if k1 has expired (older than 5 minutes)
|
// Persist the locale choice before the auth round-trip — once we
|
||||||
$k1CreatedAt = now()->subMinutes(5);
|
// navigate, this component is unmounted and $currentLangCountry
|
||||||
if ($this->k1 && now()->diffInMinutes($k1CreatedAt) >= 5) {
|
// would otherwise be lost.
|
||||||
$this->authError = 'Session expired. Please try again.';
|
session(['lang_country' => $this->currentLangCountry]);
|
||||||
|
|
||||||
return true;
|
// Hand the full-page navigation off to the client: returning a
|
||||||
}
|
// Livewire redirect from inside wire:poll has shown races with
|
||||||
|
// subsequent poll ticks (visible as a "request loop without
|
||||||
return true;
|
// redirect" for the user). Dispatching an event lets Alpine pause
|
||||||
|
// wire:poll via lightningLoginInProgress and run a clean
|
||||||
|
// window.location navigation. The dedicated /auth/complete-lightning
|
||||||
|
// controller then performs auth()->login() on a non-Livewire
|
||||||
|
// request, avoiding the session-id/CSRF rotation race that
|
||||||
|
// previously yielded 419s.
|
||||||
|
$this->dispatch(
|
||||||
|
'lightning-login-ready',
|
||||||
|
url: route('auth.ln.complete', ['k1' => $this->k1]),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -345,7 +338,8 @@ class extends Component {
|
|||||||
|
|
||||||
<div class="flex min-h-screen" x-data="nostrLogin"
|
<div class="flex min-h-screen" x-data="nostrLogin"
|
||||||
x-init="initErrorPolling"
|
x-init="initErrorPolling"
|
||||||
x-effect="document.body.style.overflow = nostrLoginInProgress ? 'hidden' : ''"
|
x-effect="document.body.style.overflow = (nostrLoginInProgress || lightningLoginInProgress) ? 'hidden' : ''"
|
||||||
|
@lightning-login-ready.window="lightningLoginInProgress = true; window.location.href = $event.detail.url"
|
||||||
data-nostr-challenge="{{ $nostrChallenge ?? '' }}">
|
data-nostr-challenge="{{ $nostrChallenge ?? '' }}">
|
||||||
<div class="flex-1 flex justify-center items-center">
|
<div class="flex-1 flex justify-center items-center">
|
||||||
<div class="w-80 max-w-80 space-y-6">
|
<div class="w-80 max-w-80 space-y-6">
|
||||||
@@ -447,7 +441,12 @@ class extends Component {
|
|||||||
flight. Otherwise wire:poll can fire a parallel /livewire/update
|
flight. Otherwise wire:poll can fire a parallel /livewire/update
|
||||||
request that races with auth()->login()'s session migration and
|
request that races with auth()->login()'s session migration and
|
||||||
lands on an invalidated session id, producing 419 TokenMismatch. --}}
|
lands on an invalidated session id, producing 419 TokenMismatch. --}}
|
||||||
<template x-if="!nostrLoginInProgress">
|
{{-- Pause Livewire polling while either login flow is mid-flight.
|
||||||
|
Otherwise wire:poll can fire a parallel /livewire/update request
|
||||||
|
that races with the navigation handled in nostrLogin.js for Nostr
|
||||||
|
or the controller-driven Lightning flow, yielding the "request loop
|
||||||
|
without redirect" symptom seen in production. --}}
|
||||||
|
<template x-if="!nostrLoginInProgress && !lightningLoginInProgress">
|
||||||
<div wire:poll.4s="checkAuth" wire:key="checkAuth"></div>
|
<div wire:poll.4s="checkAuth" wire:key="checkAuth"></div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
|||||||
@@ -83,7 +83,12 @@ class extends Component {
|
|||||||
'visible_on_map' => ['boolean'],
|
'visible_on_map' => ['boolean'],
|
||||||
]);
|
]);
|
||||||
|
|
||||||
$meetup = Meetup::create($validated);
|
$meetup = Meetup::create($validated + ['created_by' => auth()->id()]);
|
||||||
|
|
||||||
|
// Attach the creator to meetup_user so they appear under "My-Meetups"
|
||||||
|
// and pass the new edit-permission check (which is based on this pivot,
|
||||||
|
// not on created_by).
|
||||||
|
$meetup->users()->attach(auth()->id());
|
||||||
|
|
||||||
if ($this->logo) {
|
if ($this->logo) {
|
||||||
$meetup
|
$meetup
|
||||||
|
|||||||
@@ -84,13 +84,23 @@ class extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Enforce that only the meetup's creator may load or update this view.
|
* Enforce that only users who have added the meetup to their personal
|
||||||
* Mirrors services/edit and lecturer-edit. Removing this guard reopens
|
* "My-Meetups" list (the meetup_user pivot) may load or update this view.
|
||||||
* the IDOR closed by 90835f8 (security: critical fixes / edit authz).
|
* Editing is intentionally not restricted to the original `created_by`
|
||||||
|
* — any member of the meetup's user list is treated as an editor.
|
||||||
*/
|
*/
|
||||||
protected function authorizeAccess(): void
|
protected function authorizeAccess(): void
|
||||||
{
|
{
|
||||||
if (! is_null($this->meetup->created_by) && auth()->id() !== $this->meetup->created_by) {
|
if (! auth()->check()) {
|
||||||
|
abort(403);
|
||||||
|
}
|
||||||
|
|
||||||
|
$isMember = $this->meetup
|
||||||
|
->users()
|
||||||
|
->whereKey(auth()->id())
|
||||||
|
->exists();
|
||||||
|
|
||||||
|
if (! $isMember) {
|
||||||
abort(403);
|
abort(403);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,32 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
use App\Models\City;
|
||||||
|
use App\Models\Country;
|
||||||
|
|
||||||
|
beforeEach(function () {
|
||||||
|
$country = Country::factory()->create(['code' => 'de']);
|
||||||
|
City::factory()->create(['country_id' => $country->id]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('renders the sidebar with the user profile reachable on mobile viewport', function () {
|
||||||
|
actingAsUser(['name' => 'Sidebar Tester']);
|
||||||
|
|
||||||
|
$page = visit('/de/dashboard');
|
||||||
|
|
||||||
|
$page->resize(390, 844)
|
||||||
|
->click('[aria-label="Menü öffnen"]')
|
||||||
|
->assertSee('Dashboard')
|
||||||
|
->assertSee('Repository')
|
||||||
|
->assertSee('Sidebar Tester');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('renders the sidebar with the user profile on desktop viewport', function () {
|
||||||
|
actingAsUser(['name' => 'Sidebar Tester']);
|
||||||
|
|
||||||
|
$page = visit('/de/dashboard');
|
||||||
|
|
||||||
|
$page->resize(1280, 800)
|
||||||
|
->assertSee('Dashboard')
|
||||||
|
->assertSee('Repository')
|
||||||
|
->assertSee('Sidebar Tester');
|
||||||
|
});
|
||||||
@@ -105,7 +105,7 @@ it('returns 404 when the k1 path parameter is malformed', function () {
|
|||||||
->assertNotFound();
|
->assertNotFound();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('redirects auth.login checkAuth() to the completion URL without rotating the session', function () {
|
it('dispatches lightning-login-ready from auth.login checkAuth() without rotating the session', function () {
|
||||||
$user = User::factory()->create();
|
$user = User::factory()->create();
|
||||||
$k1 = bin2hex(random_bytes(32));
|
$k1 = bin2hex(random_bytes(32));
|
||||||
LoginKey::factory()->create([
|
LoginKey::factory()->create([
|
||||||
@@ -117,10 +117,41 @@ it('redirects auth.login checkAuth() to the completion URL without rotating the
|
|||||||
Livewire::test('auth.login')
|
Livewire::test('auth.login')
|
||||||
->set('k1', $k1)
|
->set('k1', $k1)
|
||||||
->call('checkAuth')
|
->call('checkAuth')
|
||||||
->assertRedirect(route('auth.ln.complete', ['k1' => $k1]));
|
->assertDispatched('lightning-login-ready', url: route('auth.ln.complete', ['k1' => $k1]));
|
||||||
|
|
||||||
// The poll handler must NOT log the user in directly — that's the
|
// The poll handler must NOT log the user in directly — that's the
|
||||||
// controller's job. Logging in here would rotate the session id and
|
// controller's job. Logging in here would rotate the session id and
|
||||||
// CSRF token mid-poll, producing 419s on any in-flight Livewire request.
|
// CSRF token mid-poll, producing 419s on any in-flight Livewire request.
|
||||||
|
// It also must NOT return a server-side redirect: emitting an event lets
|
||||||
|
// Alpine pause wire:poll via lightningLoginInProgress before navigating,
|
||||||
|
// which avoids the "request loop without redirect" symptom in production.
|
||||||
|
expect(auth()->check())->toBeFalse();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not dispatch lightning-login-ready when no LoginKey exists', function () {
|
||||||
|
$k1 = bin2hex(random_bytes(32));
|
||||||
|
|
||||||
|
Livewire::test('auth.login')
|
||||||
|
->set('k1', $k1)
|
||||||
|
->call('checkAuth')
|
||||||
|
->assertNotDispatched('lightning-login-ready');
|
||||||
|
|
||||||
|
expect(auth()->check())->toBeFalse();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not dispatch lightning-login-ready when the LoginKey is older than 5 minutes', function () {
|
||||||
|
$user = User::factory()->create();
|
||||||
|
$k1 = bin2hex(random_bytes(32));
|
||||||
|
LoginKey::factory()->create([
|
||||||
|
'user_id' => $user->id,
|
||||||
|
'k1' => $k1,
|
||||||
|
'created_at' => now()->subMinutes(10),
|
||||||
|
]);
|
||||||
|
|
||||||
|
Livewire::test('auth.login')
|
||||||
|
->set('k1', $k1)
|
||||||
|
->call('checkAuth')
|
||||||
|
->assertNotDispatched('lightning-login-ready');
|
||||||
|
|
||||||
expect(auth()->check())->toBeFalse();
|
expect(auth()->check())->toBeFalse();
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ use App\Models\City;
|
|||||||
use App\Models\Country;
|
use App\Models\Country;
|
||||||
use App\Models\Meetup;
|
use App\Models\Meetup;
|
||||||
use App\Models\MeetupEvent;
|
use App\Models\MeetupEvent;
|
||||||
|
use App\Models\User;
|
||||||
use Livewire\Livewire;
|
use Livewire\Livewire;
|
||||||
|
|
||||||
beforeEach(function () {
|
beforeEach(function () {
|
||||||
@@ -29,22 +30,42 @@ it('mounts meetups.create when authenticated', function () {
|
|||||||
Livewire::test('meetups.create')->assertStatus(200);
|
Livewire::test('meetups.create')->assertStatus(200);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('mounts meetups.edit when authenticated as the meetup creator', function () {
|
it('mounts meetups.edit when the authenticated user has added the meetup to My-Meetups', function () {
|
||||||
$owner = actingAsUser();
|
$owner = actingAsUser();
|
||||||
$meetup = Meetup::factory()->create([
|
$meetup = Meetup::factory()->create(['city_id' => $this->city->id]);
|
||||||
'city_id' => $this->city->id,
|
$meetup->users()->attach($owner);
|
||||||
'created_by' => $owner->id,
|
|
||||||
]);
|
|
||||||
|
|
||||||
Livewire::test('meetups.edit', ['meetup' => $meetup])->assertStatus(200);
|
Livewire::test('meetups.edit', ['meetup' => $meetup])->assertStatus(200);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('aborts meetups.edit with 403 when authenticated user is not the creator', function () {
|
it('mounts meetups.edit for a My-Meetups member even if another user created the meetup', function () {
|
||||||
|
$creator = User::factory()->create();
|
||||||
|
$member = actingAsUser();
|
||||||
|
$meetup = Meetup::factory()->create([
|
||||||
|
'city_id' => $this->city->id,
|
||||||
|
'created_by' => $creator->id,
|
||||||
|
]);
|
||||||
|
$meetup->users()->attach($member);
|
||||||
|
|
||||||
|
Livewire::test('meetups.edit', ['meetup' => $meetup])->assertStatus(200);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('aborts meetups.edit with 403 when the authenticated user has not added the meetup to My-Meetups', function () {
|
||||||
actingAsUser();
|
actingAsUser();
|
||||||
|
|
||||||
Livewire::test('meetups.edit', ['meetup' => $this->meetup])->assertStatus(403);
|
Livewire::test('meetups.edit', ['meetup' => $this->meetup])->assertStatus(403);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('aborts meetups.edit with 403 when the authenticated user is only the creator but not in My-Meetups', function () {
|
||||||
|
$creator = actingAsUser();
|
||||||
|
$meetup = Meetup::factory()->create([
|
||||||
|
'city_id' => $this->city->id,
|
||||||
|
'created_by' => $creator->id,
|
||||||
|
]);
|
||||||
|
|
||||||
|
Livewire::test('meetups.edit', ['meetup' => $meetup])->assertStatus(403);
|
||||||
|
});
|
||||||
|
|
||||||
it('mounts meetups.create-edit-events for new event', function () {
|
it('mounts meetups.create-edit-events for new event', function () {
|
||||||
actingAsUser();
|
actingAsUser();
|
||||||
Livewire::test('meetups.create-edit-events', ['meetup' => $this->meetup])->assertStatus(200);
|
Livewire::test('meetups.create-edit-events', ['meetup' => $this->meetup])->assertStatus(200);
|
||||||
|
|||||||
@@ -10,13 +10,13 @@ beforeEach(function () {
|
|||||||
$this->city = City::factory()->create(['country_id' => $country->id]);
|
$this->city = City::factory()->create(['country_id' => $country->id]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('updates an existing Meetup name when authenticated', function () {
|
it('updates an existing Meetup name when the user has it in My-Meetups', function () {
|
||||||
$owner = actingAsUser();
|
$member = actingAsUser();
|
||||||
$meetup = Meetup::factory()->create([
|
$meetup = Meetup::factory()->create([
|
||||||
'city_id' => $this->city->id,
|
'city_id' => $this->city->id,
|
||||||
'name' => 'Original Name',
|
'name' => 'Original Name',
|
||||||
'created_by' => $owner->id,
|
|
||||||
]);
|
]);
|
||||||
|
$meetup->users()->attach($member);
|
||||||
|
|
||||||
Livewire::test('meetups.edit', ['meetup' => $meetup])
|
Livewire::test('meetups.edit', ['meetup' => $meetup])
|
||||||
->set('name', 'Updated Name')
|
->set('name', 'Updated Name')
|
||||||
@@ -29,12 +29,12 @@ it('updates an existing Meetup name when authenticated', function () {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('rejects update when name collides with another existing Meetup', function () {
|
it('rejects update when name collides with another existing Meetup', function () {
|
||||||
$owner = actingAsUser();
|
$member = actingAsUser();
|
||||||
$meetup = Meetup::factory()->create([
|
$meetup = Meetup::factory()->create([
|
||||||
'city_id' => $this->city->id,
|
'city_id' => $this->city->id,
|
||||||
'name' => 'Original Name',
|
'name' => 'Original Name',
|
||||||
'created_by' => $owner->id,
|
|
||||||
]);
|
]);
|
||||||
|
$meetup->users()->attach($member);
|
||||||
Meetup::factory()->create(['name' => 'Other Name', 'city_id' => $this->city->id]);
|
Meetup::factory()->create(['name' => 'Other Name', 'city_id' => $this->city->id]);
|
||||||
|
|
||||||
Livewire::test('meetups.edit', ['meetup' => $meetup])
|
Livewire::test('meetups.edit', ['meetup' => $meetup])
|
||||||
@@ -44,12 +44,12 @@ it('rejects update when name collides with another existing Meetup', function ()
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('allows update when name is unchanged (Rule::unique ignores own id)', function () {
|
it('allows update when name is unchanged (Rule::unique ignores own id)', function () {
|
||||||
$owner = actingAsUser();
|
$member = actingAsUser();
|
||||||
$meetup = Meetup::factory()->create([
|
$meetup = Meetup::factory()->create([
|
||||||
'city_id' => $this->city->id,
|
'city_id' => $this->city->id,
|
||||||
'name' => 'Original Name',
|
'name' => 'Original Name',
|
||||||
'created_by' => $owner->id,
|
|
||||||
]);
|
]);
|
||||||
|
$meetup->users()->attach($member);
|
||||||
|
|
||||||
Livewire::test('meetups.edit', ['meetup' => $meetup])
|
Livewire::test('meetups.edit', ['meetup' => $meetup])
|
||||||
->set('name', 'Original Name')
|
->set('name', 'Original Name')
|
||||||
@@ -58,6 +58,19 @@ it('allows update when name is unchanged (Rule::unique ignores own id)', functio
|
|||||||
->assertHasNoErrors();
|
->assertHasNoErrors();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('blocks updateMeetup when the user has not added the meetup to My-Meetups', function () {
|
||||||
|
actingAsUser();
|
||||||
|
$meetup = Meetup::factory()->create([
|
||||||
|
'city_id' => $this->city->id,
|
||||||
|
'name' => 'Original Name',
|
||||||
|
]);
|
||||||
|
|
||||||
|
Livewire::test('meetups.edit', ['meetup' => $meetup])
|
||||||
|
->assertStatus(403);
|
||||||
|
|
||||||
|
expect($meetup->refresh()->name)->toBe('Original Name');
|
||||||
|
});
|
||||||
|
|
||||||
it('redirects guests when accessing meetup-edit', function () {
|
it('redirects guests when accessing meetup-edit', function () {
|
||||||
$meetup = Meetup::factory()->create(['city_id' => $this->city->id]);
|
$meetup = Meetup::factory()->create(['city_id' => $this->city->id]);
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user