Add Question model

This commit is contained in:
Michael Weimann 2019-12-03 20:09:37 +01:00 committed by Igor Scheller
parent 24578c5cb0
commit 4f63bbbaac
3 changed files with 258 additions and 0 deletions

64
src/Models/Question.php Normal file
View File

@ -0,0 +1,64 @@
<?php
declare(strict_types=1);
namespace Engelsystem\Models;
use Engelsystem\Models\User\User;
use Engelsystem\Models\User\UsesUserModel;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
/**
* @property integer $id
* @property string $text
* @property string $answer
* @property integer $answerer_id
* @property-read User $answerer
* @method static Builder|Question whereAnswer($value)
* @method static Builder|Question whereAnswererId($value)
* @method static Builder|Question whereId($value)
* @method static Builder|Question whereQuestion($value)
*/
class Question extends BaseModel
{
use UsesUserModel;
/** @var string[] */
protected $fillable = [
'user_id',
'text',
'answerer_id',
'answer',
];
/** @var string[] */
protected $casts = [
'user_id' => 'integer',
'answerer_id' => 'integer',
];
/**
* @return BelongsTo
*/
public function answerer(): BelongsTo
{
return $this->belongsTo(User::class, 'answerer_id');
}
/**
* @return Builder
*/
public static function unanswered(): Builder
{
return static::whereAnswererId(null);
}
/**
* @return Builder
*/
public static function answered(): Builder
{
return static::whereNotNull('answerer_id');
}
}

View File

@ -6,6 +6,7 @@ use Carbon\Carbon;
use Engelsystem\Models\BaseModel;
use Engelsystem\Models\News;
use Engelsystem\Models\NewsComment;
use Engelsystem\Models\Question;
use Illuminate\Database\Eloquent\Collection;
use Illuminate\Database\Eloquent\Relations\HasMany;
use Illuminate\Database\Eloquent\Relations\HasOne;
@ -38,6 +39,9 @@ use Illuminate\Database\Query\Builder as QueryBuilder;
* @method static QueryBuilder|User[] whereLastLoginAt($value)
* @method static QueryBuilder|User[] whereCreatedAt($value)
* @method static QueryBuilder|User[] whereUpdatedAt($value)
*
* @property-read Collection|Question[] $questionsAsked
* @property-read Collection|Question[] $questionsAnswered
*/
class User extends BaseModel
{
@ -119,4 +123,22 @@ class User extends BaseModel
{
return $this->hasMany(NewsComment::class);
}
/**
* @return HasMany
*/
public function questionsAsked(): HasMany
{
return $this->hasMany(Question::class, 'user_id')
->where('user_id', $this->id);
}
/**
* @return HasMany
*/
public function questionsAnswered(): HasMany
{
return $this->hasMany(Question::class, 'answerer_id')
->where('answerer_id', $this->id);
}
}

View File

@ -0,0 +1,172 @@
<?php
declare(strict_types=1);
namespace Unit\Models;
use Engelsystem\Models\Question;
use Engelsystem\Models\User\User;
use Engelsystem\Test\Unit\HasDatabase;
use Engelsystem\Test\Unit\TestCase;
use Illuminate\Support\Str;
class QuestionTest extends TestCase
{
use HasDatabase;
/**
* @var User
*/
private $user1;
/**
* @var User
*/
private $user2;
/**
* @return void
*/
protected function setUp(): void
{
parent::setUp();
$this->initDatabase();
$this->user1 = User::create(
[
'name' => 'user1',
'password' => '',
'email' => 'user1@example.com',
'api_key' => '',
]
);
$this->user2 = User::create(
[
'name' => 'user2',
'password' => '',
'email' => 'user2@example.com',
'api_key' => '',
]
);
}
/**
* @return void
*/
public function testStoreLoadUnAnsweredQuestion(): void
{
$question = $this->createQuestion($this->user1);
$loadedQuestion = Question::find($question->id);
$this->assertSame($this->user1->id, $loadedQuestion->user->id);
$this->assertSame($this->user1->id, $loadedQuestion->user_id);
$this->assertSame($question->text, $loadedQuestion->text);
}
/**
* @return void
*/
public function testStoreLoadAnsweredQuestion(): void
{
$question = $this->createQuestion($this->user1, $this->user2);
$loadedQuestion = Question::find($question->id);
$this->assertSame($this->user1->id, $loadedQuestion->user->id);
$this->assertSame($this->user1->id, $loadedQuestion->user_id);
$this->assertSame($loadedQuestion->text, $loadedQuestion->text);
$this->assertSame($this->user2->id, $loadedQuestion->answerer->id);
$this->assertSame($this->user2->id, $loadedQuestion->answerer_id);
$this->assertSame($loadedQuestion->answer, $loadedQuestion->answer);
}
/**
* @return void
*/
public function testUserQuestionsAsked(): void
{
$question1 = $this->createQuestion($this->user1);
$question2 = $this->createQuestion($this->user1);
// create some questions asked by user 2 to test the correct assignment
$this->createQuestion($this->user2);
$this->createQuestion($this->user2);
$user1QuestionIds = $this->user1->questionsAsked()->pluck('id')->toArray();
$this->assertCount(2, $user1QuestionIds);
$this->assertContains($question1->id, $user1QuestionIds);
$this->assertContains($question2->id, $user1QuestionIds);
}
/**
* @return void
*/
public function testUserQuestionsAnswered(): void
{
$question1 = $this->createQuestion($this->user1, $this->user2);
$question2 = $this->createQuestion($this->user1, $this->user2);
// create some questions answered by user 1 to test the correct assignment
$this->createQuestion($this->user2, $this->user1);
$this->createQuestion($this->user2, $this->user1);
$user2Answers = $this->user2->questionsAnswered()->pluck('id')->toArray();
$this->assertCount(2, $user2Answers);
$this->assertContains($question1->id, $user2Answers);
$this->assertContains($question2->id, $user2Answers);
}
/**
* @return void
*/
public function testUnanswered(): void
{
$question1 = $this->createQuestion($this->user1);
$question2 = $this->createQuestion($this->user1);
// create some answered questions as well
$this->createQuestion($this->user1, $this->user2);
$this->createQuestion($this->user1, $this->user2);
$unAnsweredQuestionIds = Question::unanswered()->pluck('id')->toArray();
$this->assertCount(2, $unAnsweredQuestionIds);
$this->assertContains($question1->id, $unAnsweredQuestionIds);
$this->assertContains($question2->id, $unAnsweredQuestionIds);
}
/**
* @return void
*/
public function testAnswered(): void
{
$question1 = $this->createQuestion($this->user1, $this->user2);
$question2 = $this->createQuestion($this->user1, $this->user2);
// create some unanswered questions as well
$this->createQuestion($this->user1);
$this->createQuestion($this->user1);
$answeredQuestionIds = Question::answered()->pluck('id')->toArray();
$this->assertCount(2, $answeredQuestionIds);
$this->assertContains($question1->id, $answeredQuestionIds);
$this->assertContains($question2->id, $answeredQuestionIds);
}
/**
* @param User $enquirer
* @param User|null $answerer
* @return Question
*/
private function createQuestion(User $enquirer, ?User $answerer = null): Question
{
$data = [
'user_id' => $enquirer->id,
'text' => Str::random(),
];
if ($answerer !== null) {
$data += [
'answerer_id' => $answerer->id,
'answer' => Str::random(),
];
}
return Question::create($data);
}
}