Merge branch 'picker-service'

This commit is contained in:
Oliver Davies 2019-02-14 01:34:55 +00:00
commit 53e69067e4
6 changed files with 312 additions and 2 deletions

View file

@ -6,6 +6,7 @@
"ext-ctype": "*",
"ext-iconv": "*",
"guzzlehttp/guzzle": "^6.3",
"josephlavin/tap": "^1.0",
"symfony/cache": "4.2.*",
"symfony/console": "4.2.*",
"symfony/dotenv": "4.2.*",

41
composer.lock generated
View file

@ -4,7 +4,7 @@
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
"This file is @generated automatically"
],
"content-hash": "15c99cfaa96233a5a12cf632ac267c27",
"content-hash": "35f8b30095e57576a83ae4bdaa898a19",
"packages": [
{
"name": "guzzlehttp/guzzle",
@ -189,6 +189,45 @@
],
"time": "2018-12-04T20:46:45+00:00"
},
{
"name": "josephlavin/tap",
"version": "v1.0.0",
"source": {
"type": "git",
"url": "https://github.com/josephlavin/tap.git",
"reference": "8733d96fb9b7d7dca204dfe59ec3ff34f3c69be9"
},
"dist": {
"type": "zip",
"url": "https://api.github.com/repos/josephlavin/tap/zipball/8733d96fb9b7d7dca204dfe59ec3ff34f3c69be9",
"reference": "8733d96fb9b7d7dca204dfe59ec3ff34f3c69be9",
"shasum": ""
},
"require": {
"php": ">=5.6.0"
},
"require-dev": {
"phpunit/phpunit": "~5.7"
},
"type": "library",
"autoload": {
"files": [
"src/tap.php",
"src/TapProxy.php"
]
},
"notification-url": "https://packagist.org/downloads/",
"license": [
"MIT"
],
"description": "Stand alone port of Laravel's tap method.",
"keywords": [
"laravel",
"php",
"tap"
],
"time": "2017-09-24T23:36:28+00:00"
},
{
"name": "psr/cache",
"version": "1.0.1",

View file

@ -2,6 +2,7 @@
namespace App\Command;
use App\Service\Picker;
use GuzzleHttp\Client;
use Symfony\Component\Cache\Simple\FilesystemCache;
use Symfony\Component\Console\Command\Command;
@ -19,10 +20,22 @@ class PickWinnerCommand extends Command
*/
private $client;
public function __construct()
/**
* @var \App\Service\Picker
*/
private $picker;
/**
* PickWinnerCommand constructor.
*
* @param \App\Service\Picker $picker
* The Picker service.
*/
public function __construct(Picker $picker)
{
parent::__construct();
$this->client = new Client();
$this->picker = $picker;
}
/**

113
src/Service/Picker.php Normal file
View file

@ -0,0 +1,113 @@
<?php
namespace App\Service;
use Tightenco\Collect\Support\Collection;
class Picker
{
/**
* The combined hosts for the retrieved events.
*
* @var \Tightenco\Collect\Support\Collection
*/
private $hosts;
/**
* @var \Tightenco\Collect\Support\Collection
*/
private $comments;
/**
* Picker constructor.
*/
public function __construct()
{
$this->comments = collect();
$this->hosts = collect();
}
/**
* Retrieve the combined comments for all events.
*
* @return \Tightenco\Collect\Support\Collection
*/
public function getComments(): Collection
{
return $this->comments;
}
/**
* Retrieve the event hosts.
*
* @return \Tightenco\Collect\Support\Collection
*/
public function getHosts(): Collection
{
return $this->hosts;
}
/**
* Set the hosts for the retrieved events.
*
* @param \Tightenco\Collect\Support\Collection $data
* The event data.
*
* @return self
*/
public function setHosts(Collection $data): self
{
$this->hosts = $data->pluck('hosts.*.host_name')
->flatten(1)
->unique()
->sort();
return $this;
}
/**
* Set the comments for the events.
*
* @param \Tightenco\Collect\Support\Collection $comments
* A collection of comments.
*
* @return $this
*/
public function setComments(Collection $comments): self
{
$this->comments = $comments
->flatten(1)
->filter(function (array $comment) {
return !$this->isUserAnEventHost($comment['user_display_name']);
})
->values();
return $this;
}
/**
* Determine whether a commenter is an event host.
*
* @param string $user_display_name
* The user's display name.
*
* @return bool
*/
private function isUserAnEventHost(string $user_display_name): bool
{
return $this->hosts->contains($user_display_name);
}
/**
* Select and return the winners.
*
* @param int $count
* The number of winners.
*
* @return \Tightenco\Collect\Support\Collection
*/
public function getWinners(int $count): Collection
{
return $this->getComments()->random($count);
}
}

View file

@ -14,6 +14,9 @@
"guzzlehttp/psr7": {
"version": "1.5.2"
},
"josephlavin/tap": {
"version": "v1.0.0"
},
"nikic/php-parser": {
"version": "v4.2.0"
},

View file

@ -0,0 +1,141 @@
<?php
namespace App\Tests\Service;
use App\Service\Picker;
use PHPUnit\Framework\TestCase;
use Tightenco\Collect\Support\Collection;
class PickerTest extends TestCase
{
/** @test */
public function hosts_for_multiple_events_are_grouped_and_unique()
{
$data = [
[
'hosts' => [
['host_name' => 'Lee Stone'],
['host_name' => 'Dave Liddament'],
['host_name' => 'Kat Zien'],
],
],
[
'hosts' => [
['host_name' => 'Oliver Davies'],
['host_name' => 'Lee Stone'],
['host_name' => 'Lucia Velasco'],
['host_name' => 'Dave Liddament'],
],
],
];
$hosts = (new Picker())
->setHosts(collect($data))
->getHosts();
$this->assertInstanceOf(Collection::class, $hosts);
$this->assertCount(5, $hosts);
}
/** @test */
public function comments_for_multiple_events_are_flattened_and_combined()
{
$data = [
[
[
'comment' => 'Great talk!',
'user_display_name' => 'Dan Ackroyd',
],
[
'comment' => 'Could be better.',
'user_display_name' => 'Lucia Velasco',
],
],
[
[
'comment' => 'Needs more cat pictures.',
'user_display_name' => 'Rupert Jabelman',
],
],
];
$comments = (new Picker())
->setComments(collect($data))
->getComments();
$this->assertInstanceOf(Collection::class, $comments);
$this->assertCount(3, $comments);
}
/** @test */
public function comments_from_event_hosts_cannot_be_picked()
{
$event = [
'hosts' => [
['host_name' => 'Oliver Davies'],
],
];
$comments = [
[
[
'comment' => 'Great talk!',
'user_display_name' => 'Peter Fisher',
],
[
'comment' => 'Text on slides could be bigger.',
'user_display_name' => 'Oliver Davies',
],
[
'comment' => 'Speak slower.',
'user_display_name' => 'Zan Baldwin',
],
],
];
$comments = (new Picker())
->setHosts(collect([$event]))
->setComments(collect($comments))
->getComments();
$this->assertCount(2, $comments);
$this->assertSame('Peter Fisher', $comments[0]['user_display_name']);
$this->assertSame('Zan Baldwin', $comments[1]['user_display_name']);
}
/** @test */
public function winners_can_be_selected()
{
$comments = [
[
[
'comment' => 'Great talk!',
'user_display_name' => 'Peter Fisher',
],
[
'comment' => 'Text on slides could be bigger.',
'user_display_name' => 'Michael Bush',
],
[
'comment' => 'Speak slower.',
'user_display_name' => 'Zan Baldwin',
],
],
];
$picker = new Picker();
$picker->setComments(collect($comments));
$this->assertCount(3, $picker->getComments());
tap($picker->getWinners(1), function (Collection $winners) use ($picker) {
$this->assertCount(1, $winners);
$this->assertTrue($picker->getComments()->contains($winners->first()));
});
tap($picker->getWinners(2), function (Collection $winners) use ($picker) {
$this->assertCount(2, $winners);
$this->assertTrue($picker->getComments()->contains($winners->first()));
$this->assertTrue($picker->getComments()->contains($winners->last()));
});
}
}