9c900d6553
CI / Tests (PHP 8.1) (pull_request) Successful in 47s
CI / No Debug Code (pull_request) Successful in 2s
CI / Build Plugin Zip (pull_request) Has been skipped
CI / Coding Standards (pull_request) Successful in 52s
CI / PHPStan (pull_request) Successful in 1m1s
CI / Tests (PHP 8.2) (pull_request) Successful in 48s
CI / Tests (PHP 8.3) (pull_request) Successful in 45s
Implements #16: invite-only student self-registration through a front-end page, accepting signup-scoped policies at account creation. Policy domain: - us_policies.acceptance_scope (signup/booking/both); Policy::appliesTo(); PolicyRepository::findForScope(); scope threaded through PolicyService, the REST create, the admin controller, and the Policies form. - PolicyAcceptance::REG_ACCOUNT (registration_id = the new user's ID). Auth: - Invite value object + InviteRepository; us_invites table. - RegistrationController + Invites admin page (manage_students): invite an email, share the registration link, revoke. - RegistrationPage ([us_student_register] shortcode): validates the invite token, collects name/password, renders signup-scoped published policies with required acceptance, creates the us_student user, records account-type acceptances, marks the invite accepted, and logs the user in. - RoleManager: manage_students cap added to STUDIO_ADMIN_CAPS. Invite-only is implemented; the us_registration_mode self_approval path is a documented future seam. Docs: docs/features/account-registration.md; policies.md updated. Tests: tests/Unit/Auth/ (Invite, InviteRepository) plus Policy scope updates. composer test (104), cs, and PHPStan level 6 all pass. Refs #16 Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
100 lines
3.8 KiB
PHP
100 lines
3.8 KiB
PHP
<?php
|
|
declare(strict_types=1);
|
|
|
|
namespace Unsupervised\Schedular\Tests\Unit\Policy;
|
|
|
|
use Unsupervised\Schedular\Policy\Policy;
|
|
use Unsupervised\Schedular\Policy\PolicyAcceptance;
|
|
use Unsupervised\Schedular\Policy\PolicyVersion;
|
|
use Unsupervised\Schedular\Tests\Unit\TestCase;
|
|
|
|
class PolicyValueObjectsTest extends TestCase
|
|
{
|
|
public function testPolicyFromRowAndToArray(): void
|
|
{
|
|
$policy = Policy::fromRow((object) [
|
|
'id' => '4',
|
|
'title' => 'Cancellation',
|
|
'slug' => 'cancellation',
|
|
'current_version_id' => '9',
|
|
'acceptance_scope' => Policy::SCOPE_BOTH,
|
|
]);
|
|
|
|
self::assertSame(4, $policy->id);
|
|
self::assertSame('cancellation', $policy->slug);
|
|
self::assertSame(9, $policy->currentVersionId);
|
|
self::assertSame(Policy::SCOPE_BOTH, $policy->acceptanceScope);
|
|
self::assertArrayHasKey('acceptance_scope', $policy->toArray());
|
|
}
|
|
|
|
public function testPolicyHandlesNullCurrentVersion(): void
|
|
{
|
|
$policy = Policy::fromRow((object) [
|
|
'id' => '4',
|
|
'title' => 'Cancellation',
|
|
'slug' => 'cancellation',
|
|
'current_version_id' => null,
|
|
'acceptance_scope' => Policy::SCOPE_BOOKING,
|
|
]);
|
|
|
|
self::assertNull($policy->currentVersionId);
|
|
}
|
|
|
|
public function testPolicyAppliesToScopeAndBoth(): void
|
|
{
|
|
$signup = new Policy('Terms', 'terms', acceptanceScope: Policy::SCOPE_SIGNUP);
|
|
self::assertTrue($signup->appliesTo(Policy::SCOPE_SIGNUP));
|
|
self::assertFalse($signup->appliesTo(Policy::SCOPE_BOOKING));
|
|
|
|
$both = new Policy('Both', 'both', acceptanceScope: Policy::SCOPE_BOTH);
|
|
self::assertTrue($both->appliesTo(Policy::SCOPE_SIGNUP));
|
|
self::assertTrue($both->appliesTo(Policy::SCOPE_BOOKING));
|
|
}
|
|
|
|
public function testPolicyVersionFromRowAndStatusHelper(): void
|
|
{
|
|
$version = PolicyVersion::fromRow((object) [
|
|
'id' => '9',
|
|
'policy_id' => '4',
|
|
'version_number' => '2',
|
|
'body' => '<p>Policy</p>',
|
|
'status' => PolicyVersion::STATUS_PUBLISHED,
|
|
'published_at' => '2026-06-01 10:00:00',
|
|
]);
|
|
|
|
self::assertSame(9, $version->id);
|
|
self::assertSame(2, $version->versionNumber);
|
|
self::assertTrue($version->isPublished());
|
|
self::assertSame('2026-06-01 10:00:00', $version->publishedAt);
|
|
}
|
|
|
|
public function testPolicyVersionDefaultsToDraft(): void
|
|
{
|
|
$version = new PolicyVersion(4, 1);
|
|
|
|
self::assertSame(PolicyVersion::STATUS_DRAFT, $version->status);
|
|
self::assertFalse($version->isPublished());
|
|
self::assertNull($version->publishedAt);
|
|
self::assertContains(PolicyVersion::STATUS_ARCHIVED, PolicyVersion::VALID_STATUSES);
|
|
}
|
|
|
|
public function testPolicyAcceptanceFromRowAndToArray(): void
|
|
{
|
|
$acceptance = PolicyAcceptance::fromRow((object) [
|
|
'id' => '1',
|
|
'policy_version_id' => '9',
|
|
'student_id' => '5',
|
|
'registration_type' => PolicyAcceptance::REG_LESSON,
|
|
'registration_id' => '12',
|
|
'accepted_at' => '2026-06-02 09:00:00',
|
|
'ip_address' => '203.0.113.7',
|
|
]);
|
|
|
|
self::assertSame(9, $acceptance->policyVersionId);
|
|
self::assertSame(PolicyAcceptance::REG_LESSON, $acceptance->registrationType);
|
|
self::assertSame('203.0.113.7', $acceptance->ipAddress);
|
|
self::assertArrayHasKey('policy_version_id', $acceptance->toArray());
|
|
self::assertContains(PolicyAcceptance::REG_ENROLLMENT, PolicyAcceptance::VALID_REGISTRATION_TYPES);
|
|
}
|
|
}
|