553 lines
18 KiB
PHP
553 lines
18 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Engelsystem\Test\Unit\Controllers\Metrics;
|
|
|
|
use Carbon\Carbon;
|
|
use Engelsystem\Controllers\Metrics\Stats;
|
|
use Engelsystem\Models\Faq;
|
|
use Engelsystem\Models\LogEntry;
|
|
use Engelsystem\Models\Message;
|
|
use Engelsystem\Models\News;
|
|
use Engelsystem\Models\NewsComment;
|
|
use Engelsystem\Models\OAuth;
|
|
use Engelsystem\Models\Question;
|
|
use Engelsystem\Models\Room;
|
|
use Engelsystem\Models\Shifts\Shift;
|
|
use Engelsystem\Models\Shifts\ShiftEntry;
|
|
use Engelsystem\Models\User\License;
|
|
use Engelsystem\Models\User\PasswordReset;
|
|
use Engelsystem\Models\User\PersonalData;
|
|
use Engelsystem\Models\User\Settings;
|
|
use Engelsystem\Models\User\State;
|
|
use Engelsystem\Models\User\User;
|
|
use Engelsystem\Models\Worklog;
|
|
use Engelsystem\Test\Unit\HasDatabase;
|
|
use Engelsystem\Test\Unit\TestCase;
|
|
use Illuminate\Support\Str;
|
|
use Psr\Log\LogLevel;
|
|
|
|
class StatsTest extends TestCase
|
|
{
|
|
use HasDatabase;
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::__construct
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::newUsers
|
|
*/
|
|
public function testNewUsers(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(2, $stats->newUsers());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::vouchers
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::vouchersQuery
|
|
*/
|
|
public function testVouchers(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(14, $stats->vouchers());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::vouchersBuckets
|
|
*/
|
|
public function testVouchersBuckets(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals([1 => 6, 3 => 8, '+Inf' => 9], $stats->vouchersBuckets([1, 3, '+Inf']));
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::tshirts
|
|
*/
|
|
public function testTshirts(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(2, $stats->tshirts());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::tshirtSizes
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::raw
|
|
*/
|
|
public function testTshirtSizes(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
$stats = new Stats($this->database);
|
|
$sizes = $stats->tshirtSizes();
|
|
$this->assertCount(2, $sizes);
|
|
$this->assertEquals([
|
|
['shirt_size' => 'L', 'count' => 2],
|
|
['shirt_size' => 'XXL', 'count' => 1],
|
|
], $sizes->toArray());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::languages
|
|
*/
|
|
public function testLanguages(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
$stats = new Stats($this->database);
|
|
$languages = $stats->languages();
|
|
$this->assertCount(2, $languages);
|
|
$this->assertEquals([
|
|
['language' => 'lo_RM', 'count' => 2],
|
|
['language' => 'te_ST', 'count' => 7],
|
|
], $languages->toArray());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::themes
|
|
*/
|
|
public function testThemes(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
$stats = new Stats($this->database);
|
|
$themes = $stats->themes();
|
|
$this->assertCount(3, $themes);
|
|
$this->assertEquals([
|
|
['theme' => 0, 'count' => 6],
|
|
['theme' => 1, 'count' => 2],
|
|
['theme' => 4, 'count' => 1],
|
|
], $themes->toArray());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::licenses
|
|
*/
|
|
public function testLicenses(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(1, $stats->licenses('has_car'));
|
|
$this->assertEquals(1, $stats->licenses('forklift'));
|
|
$this->assertEquals(2, $stats->licenses('car'));
|
|
$this->assertEquals(0, $stats->licenses('3.5t'));
|
|
$this->assertEquals(0, $stats->licenses('7.5t'));
|
|
$this->assertEquals(1, $stats->licenses('12t'));
|
|
$this->assertEquals(0, $stats->licenses('ifsg_light'));
|
|
$this->assertEquals(0, $stats->licenses('ifsg'));
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::worklogSeconds
|
|
*/
|
|
public function testWorklogSeconds(): void
|
|
{
|
|
$this->addUsers();
|
|
$worklogData = [
|
|
'user_id' => 1,
|
|
'creator_id' => 1,
|
|
'hours' => 2.4,
|
|
'comment' => '',
|
|
'worked_at' => new Carbon(),
|
|
];
|
|
(new Worklog($worklogData))->save();
|
|
(new Worklog(['hours' => 1.2, 'user_id' => 3] + $worklogData))->save();
|
|
|
|
$stats = new Stats($this->database);
|
|
$seconds = $stats->worklogSeconds();
|
|
|
|
$this->assertEquals(2.4 * 60 * 60 + 1.2 * 60 * 60, $seconds);
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::worklogBuckets
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::getBuckets
|
|
*/
|
|
public function testWorklogBuckets(): void
|
|
{
|
|
Worklog::factory()->create(['hours' => 1.2, 'worked_at' => Carbon::now()->subDay()]);
|
|
Worklog::factory()->create(['hours' => 1.9, 'worked_at' => Carbon::now()->subDay()]);
|
|
Worklog::factory()->create(['hours' => 3, 'worked_at' => Carbon::now()->subDay()]);
|
|
Worklog::factory()->create(['hours' => 10, 'worked_at' => Carbon::now()->subDay()]);
|
|
|
|
$stats = new Stats($this->database);
|
|
$buckets = $stats->worklogBuckets([
|
|
1 * 60 * 60,
|
|
2 * 60 * 60,
|
|
3 * 60 * 60,
|
|
4 * 60 * 60,
|
|
'+Inf',
|
|
]);
|
|
|
|
$this->assertEquals([
|
|
3600 => 0,
|
|
7200 => 2,
|
|
10800 => 3,
|
|
14400 => 3,
|
|
'+Inf' => 4,
|
|
], $buckets);
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::rooms
|
|
*/
|
|
public function testRooms(): void
|
|
{
|
|
(new Room(['name' => 'Location 1']))->save();
|
|
(new Room(['name' => 'Second location']))->save();
|
|
(new Room(['name' => 'Another location']))->save();
|
|
(new Room(['name' => 'Old location']))->save();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(4, $stats->rooms());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::shifts
|
|
*/
|
|
public function testShifts(): void
|
|
{
|
|
Shift::factory(5)->create();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(5, $stats->shifts());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::announcements
|
|
*/
|
|
public function testAnnouncements(): void
|
|
{
|
|
$this->addUsers();
|
|
$newsData = ['title' => 'Test', 'text' => 'Foo Bar', 'user_id' => 1];
|
|
|
|
(new News($newsData))->save();
|
|
(new News($newsData))->save();
|
|
(new News($newsData + ['is_meeting' => true]))->save();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(3, $stats->announcements());
|
|
$this->assertEquals(2, $stats->announcements(false));
|
|
$this->assertEquals(1, $stats->announcements(true));
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::comments
|
|
*/
|
|
public function testComments(): void
|
|
{
|
|
$user = $this->addUser();
|
|
|
|
$news = new News(['title' => 'Test', 'text' => 'Foo Bar', 'user_id' => $user->id]);
|
|
$news->save();
|
|
|
|
foreach (['Test', 'Another text!'] as $text) {
|
|
$comment = new NewsComment(['text' => $text]);
|
|
$comment->news()->associate($news);
|
|
$comment->user()->associate($user);
|
|
$comment->save();
|
|
}
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(2, $stats->comments());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::questions
|
|
*/
|
|
public function testQuestions(): void
|
|
{
|
|
$this->addUsers();
|
|
$questionsData = ['text' => 'Lorem Ipsum', 'user_id' => 1];
|
|
|
|
(new Question($questionsData))->save();
|
|
(new Question($questionsData))->save();
|
|
(new Question($questionsData + ['answerer_id' => 2, 'answer' => 'Dolor sit!']))->save();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(3, $stats->questions());
|
|
$this->assertEquals(2, $stats->questions(false));
|
|
$this->assertEquals(1, $stats->questions(true));
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::arrivedUsers
|
|
*/
|
|
public function testArrivedUsers(): void
|
|
{
|
|
$this->addUsers();
|
|
ShiftEntry::factory()->create(['user_id' => 3]);
|
|
ShiftEntry::factory()->create(['user_id' => 4]);
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(7, $stats->arrivedUsers());
|
|
$this->assertEquals(5, $stats->arrivedUsers(false));
|
|
$this->assertEquals(2, $stats->arrivedUsers(true));
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::forceActiveUsers
|
|
*/
|
|
public function testForceActiveUsers(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(2, $stats->forceActiveUsers());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::usersPronouns
|
|
*/
|
|
public function testUsersPronouns(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(2, $stats->usersPronouns());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::email
|
|
*/
|
|
public function testEmail(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(0, $stats->email('not-available-option'));
|
|
$this->assertEquals(2, $stats->email('system'));
|
|
$this->assertEquals(3, $stats->email('humans'));
|
|
$this->assertEquals(1, $stats->email('goody'));
|
|
$this->assertEquals(1, $stats->email('news'));
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::currentlyWorkingUsers
|
|
*/
|
|
public function testCurrentlyWorkingUsers(): void
|
|
{
|
|
$this->addUsers();
|
|
/** @var Shift $shift */
|
|
$shift = Shift::factory()->create(['start' => Carbon::now()->subHour(), 'end' => Carbon::now()->addHour()]);
|
|
|
|
ShiftEntry::factory()->create(['shift_id' => $shift->id, 'freeloaded' => false]);
|
|
ShiftEntry::factory()->create(['shift_id' => $shift->id, 'freeloaded' => false]);
|
|
ShiftEntry::factory()->create(['shift_id' => $shift->id, 'freeloaded' => true]);
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(3, $stats->currentlyWorkingUsers());
|
|
$this->assertEquals(2, $stats->currentlyWorkingUsers(false));
|
|
$this->assertEquals(1, $stats->currentlyWorkingUsers(true));
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::faq
|
|
*/
|
|
public function testFaq(): void
|
|
{
|
|
(new Faq(['question' => 'Foo?', 'text' => 'Bar!']))->save();
|
|
(new Faq(['question' => 'Lorem??', 'text' => 'Ipsum!!!']))->save();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(2, $stats->faq());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::messages
|
|
*/
|
|
public function testMessages(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
(new Message(['user_id' => 1, 'receiver_id' => 2, 'text' => 'Ohi?']))->save();
|
|
(new Message(['user_id' => 4, 'receiver_id' => 1, 'text' => 'Testing stuff?']))->save();
|
|
(new Message(['user_id' => 2, 'receiver_id' => 3, 'text' => 'Nope!', 'read' => true]))->save();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(3, $stats->messages());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::sessions
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::getQuery
|
|
*/
|
|
public function testSessions(): void
|
|
{
|
|
$this->database
|
|
->getConnection()
|
|
->table('sessions')
|
|
->insert([
|
|
['id' => 'asd', 'payload' => 'data', 'last_activity' => new Carbon('1 month ago')],
|
|
['id' => 'efg', 'payload' => 'lorem', 'last_activity' => new Carbon('55 minutes ago')],
|
|
['id' => 'hij', 'payload' => 'ipsum', 'last_activity' => new Carbon('3 seconds ago')],
|
|
['id' => 'klm', 'payload' => 'dolor', 'last_activity' => new Carbon()],
|
|
]);
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(4, $stats->sessions());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::oauth
|
|
*/
|
|
public function testOauth(): void
|
|
{
|
|
$this->addUsers();
|
|
$user1 = User::find(1);
|
|
$user2 = User::find(2);
|
|
$user3 = User::find(3);
|
|
|
|
(new OAuth(['provider' => 'test', 'identifier' => '1']))->user()->associate($user1)->save();
|
|
(new OAuth(['provider' => 'test', 'identifier' => '2']))->user()->associate($user2)->save();
|
|
(new OAuth(['provider' => 'another-provider', 'identifier' => 'usr3']))->user()->associate($user3)->save();
|
|
|
|
$stats = new Stats($this->database);
|
|
$oauth = $stats->oauth();
|
|
|
|
$this->assertCount(2, $oauth);
|
|
$this->assertEquals([
|
|
['provider' => 'another-provider', 'count' => 1],
|
|
['provider' => 'test', 'count' => 2],
|
|
], $oauth->toArray());
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::databaseRead
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::databaseWrite
|
|
*/
|
|
public function testDatabase(): void
|
|
{
|
|
$stats = new Stats($this->database);
|
|
|
|
$read = $stats->databaseRead();
|
|
$write = $stats->databaseWrite();
|
|
|
|
$this->assertIsFloat($read);
|
|
$this->assertNotEmpty($read);
|
|
$this->assertIsFloat($write);
|
|
$this->assertNotEmpty($write);
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::logEntries
|
|
*/
|
|
public function testLogEntries(): void
|
|
{
|
|
(new LogEntry(['level' => LogLevel::INFO, 'message' => 'Some info']))->save();
|
|
(new LogEntry(['level' => LogLevel::INFO, 'message' => 'Another info']))->save();
|
|
(new LogEntry(['level' => LogLevel::CRITICAL, 'message' => 'A critical error!']))->save();
|
|
(new LogEntry(['level' => LogLevel::DEBUG, 'message' => 'Verbose output!']))->save();
|
|
(new LogEntry(['level' => LogLevel::INFO, 'message' => 'Shutdown initiated']))->save();
|
|
(new LogEntry(['level' => LogLevel::WARNING, 'message' => 'Please be cautious']))->save();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(6, $stats->logEntries());
|
|
$this->assertEquals(3, $stats->logEntries(LogLevel::INFO));
|
|
$this->assertEquals(1, $stats->logEntries(LogLevel::DEBUG));
|
|
}
|
|
|
|
/**
|
|
* @covers \Engelsystem\Controllers\Metrics\Stats::passwordResets
|
|
*/
|
|
public function testPasswordResets(): void
|
|
{
|
|
$this->addUsers();
|
|
|
|
(new PasswordReset(['user_id' => 1, 'token' => 'loremIpsum123']))->save();
|
|
(new PasswordReset(['user_id' => 3, 'token' => '5omeR4nd0mTok3N']))->save();
|
|
|
|
$stats = new Stats($this->database);
|
|
$this->assertEquals(2, $stats->passwordResets());
|
|
}
|
|
|
|
/**
|
|
* Add some example users
|
|
*/
|
|
protected function addUsers(): void
|
|
{
|
|
$this->addUser();
|
|
$this->addUser([], ['shirt_size' => 'L'], ['email_human' => true, 'email_shiftinfo' => true]);
|
|
$this->addUser(['arrived' => 1], [], ['email_human' => true, 'email_goody' => true, 'email_news' => true]);
|
|
$this->addUser(['arrived' => 1], ['pronoun' => 'unicorn'], ['language' => 'lo_RM', 'email_shiftinfo' => true]);
|
|
$this->addUser(['arrived' => 1, 'got_voucher' => 2], ['shirt_size' => 'XXL'], ['language' => 'lo_RM']);
|
|
$this->addUser(
|
|
['arrived' => 1, 'got_voucher' => 9, 'force_active' => true],
|
|
[],
|
|
['theme' => 1],
|
|
['drive_car' => true, 'drive_12t' => true]
|
|
);
|
|
$this->addUser(
|
|
['arrived' => 1, 'got_voucher' => 3],
|
|
['pronoun' => 'per'],
|
|
['theme' => 1, 'email_human' => true],
|
|
['has_car' => true, 'drive_forklift' => true, 'drive_car' => true]
|
|
);
|
|
$this->addUser(['arrived' => 1, 'active' => 1, 'got_shirt' => true, 'force_active' => true]);
|
|
$this->addUser(['arrived' => 1, 'active' => 1, 'got_shirt' => true], ['shirt_size' => 'L'], ['theme' => 4]);
|
|
}
|
|
|
|
protected function addUser(
|
|
array $state = [],
|
|
array $personalData = [],
|
|
array $settings = [],
|
|
array $license = []
|
|
): User {
|
|
$name = 'user_' . Str::random(5);
|
|
|
|
$user = new User([
|
|
'name' => $name,
|
|
'password' => '',
|
|
'email' => $name . '@engel.example.com',
|
|
'api_key' => '',
|
|
]);
|
|
$user->save();
|
|
|
|
$state = new State($state);
|
|
$state->user()
|
|
->associate($user)
|
|
->save();
|
|
|
|
$personalData = new PersonalData($personalData);
|
|
$personalData->user()
|
|
->associate($user)
|
|
->save();
|
|
|
|
$settings = new Settings(array_merge([
|
|
'language' => 'te_ST',
|
|
'theme' => 0,
|
|
'email_human' => false,
|
|
'email_shiftinfo' => false,
|
|
], $settings));
|
|
$settings->user()
|
|
->associate($user)
|
|
->save();
|
|
|
|
$license = new License($license);
|
|
$license->user()
|
|
->associate($user)
|
|
->save();
|
|
|
|
return $user;
|
|
}
|
|
|
|
/**
|
|
* Set up the environment
|
|
*/
|
|
protected function setUp(): void
|
|
{
|
|
parent::setUp();
|
|
|
|
$this->initDatabase();
|
|
}
|
|
}
|