0
0
Fork 0
mirror of https://github.com/kevinpapst/kimai2.git synced 2025-01-11 03:48:10 +00:00
kevinpapst_kimai2/tests/Invoice/ServiceInvoiceTest.php
2024-12-22 01:25:30 +01:00

327 lines
12 KiB
PHP

<?php
/*
* This file is part of the Kimai time-tracking app.
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
namespace App\Tests\Invoice;
use App\Configuration\LocaleService;
use App\Entity\Customer;
use App\Entity\Invoice;
use App\Entity\InvoiceTemplate;
use App\Entity\Project;
use App\Entity\Timesheet;
use App\Invoice\Calculator\DefaultCalculator;
use App\Invoice\InvoiceItemRepositoryInterface;
use App\Invoice\InvoiceModel;
use App\Invoice\NumberGenerator\DateNumberGenerator;
use App\Invoice\Renderer\TwigRenderer;
use App\Invoice\ServiceInvoice;
use App\Model\InvoiceDocument;
use App\Repository\InvoiceDocumentRepository;
use App\Repository\InvoiceRepository;
use App\Repository\Query\InvoiceQuery;
use App\Tests\Mocks\InvoiceModelFactoryFactory;
use App\Utils\FileHelper;
use PHPUnit\Framework\TestCase;
use Twig\Environment;
/**
* @covers \App\Invoice\ServiceInvoice
*/
class ServiceInvoiceTest extends TestCase
{
private function getSut(array $paths): ServiceInvoice
{
$languages = [
'en' => LocaleService::DEFAULT_SETTINGS
];
$formattings = new LocaleService($languages);
$repo = new InvoiceDocumentRepository($paths);
$invoiceRepo = $this->createMock(InvoiceRepository::class);
return new ServiceInvoice($repo, new FileHelper(realpath(__DIR__ . '/../../var/data/')), $invoiceRepo, $formattings, (new InvoiceModelFactoryFactory($this))->create());
}
public function testInvalidExceptionOnChangeState(): void
{
$this->expectException(\InvalidArgumentException::class);
$this->expectExceptionMessage('Unknown invoice status');
$sut = $this->getSut([]);
$sut->changeInvoiceStatus(new Invoice(), 'foo');
}
public function testEmptyObject(): void
{
$sut = $this->getSut([]);
self::assertEmpty($sut->getCalculator());
self::assertIsArray($sut->getCalculator());
self::assertEmpty($sut->getRenderer());
self::assertIsArray($sut->getRenderer());
self::assertEmpty($sut->getNumberGenerator());
self::assertIsArray($sut->getNumberGenerator());
self::assertEmpty($sut->getDocuments());
self::assertIsArray($sut->getDocuments());
self::assertNull($sut->getCalculatorByName('default'));
self::assertNull($sut->getDocumentByName('default'));
self::assertNull($sut->getNumberGeneratorByName('default'));
}
public function testWithDocumentDirectory(): void
{
$sut = $this->getSut(['templates/invoice/renderer/']);
$actual = $sut->getDocuments();
self::assertNotEmpty($actual);
$actual = $sut->getDocumentByName('default');
self::assertInstanceOf(InvoiceDocument::class, $actual);
}
public function testAdd(): void
{
$sut = $this->getSut([]);
$sut->addCalculator(new DefaultCalculator());
$sut->addNumberGenerator($this->getNumberGeneratorSut());
$twig = $this->getMockBuilder(Environment::class)->disableOriginalConstructor()->getMock();
$sut->addRenderer(new TwigRenderer($twig));
self::assertEquals(1, \count($sut->getCalculator()));
self::assertInstanceOf(DefaultCalculator::class, $sut->getCalculatorByName('default'));
self::assertEquals(1, \count($sut->getNumberGenerator()));
self::assertInstanceOf(DateNumberGenerator::class, $sut->getNumberGeneratorByName('date'));
self::assertEquals(1, \count($sut->getRenderer()));
}
public function testCreateModelThrowsOnMissingTemplate(): void
{
$this->expectException(\Exception::class);
$this->expectExceptionMessage('Cannot create invoice model without template');
$query = new InvoiceQuery();
$query->setCustomers([new Customer('foo')]);
$sut = $this->getSut([]);
$sut->createModel($query);
}
public function testCreateModelUsesTemplateLanguage(): void
{
$template = new InvoiceTemplate();
$template->setNumberGenerator('date');
$template->setLanguage('it');
$query = new InvoiceQuery();
$query->setCustomers([new Customer('foo')]);
$query->setTemplate($template);
$sut = $this->getSut([]);
$sut->addCalculator(new DefaultCalculator());
$sut->addNumberGenerator($this->getNumberGeneratorSut());
$model = $sut->createModel($query);
self::assertEquals('it', $model->getTemplate()->getLanguage());
}
public function testBeginAndEndDateFallback(): void
{
$timezone = new \DateTimeZone('Europe/Vienna');
$customer = new Customer('foo');
$project = new Project();
$project->setCustomer($customer);
$timesheet1 = new Timesheet();
$timesheet1->setProject($project);
$timesheet1->setBegin(new \DateTime('2011-01-27 12:12:12', $timezone));
$timesheet1->setEnd(new \DateTime('2020-01-27 12:12:12', $timezone));
$timesheet2 = new Timesheet();
$timesheet2->setProject($project);
$timesheet2->setBegin(new \DateTime('2010-01-27 08:24:33', $timezone));
$timesheet2->setEnd(new \DateTime('2019-01-27 12:12:12', $timezone));
$timesheet3 = new Timesheet();
$timesheet3->setProject($project);
$timesheet3->setBegin(new \DateTime('2019-01-27 12:12:12', $timezone));
$timesheet3->setEnd(new \DateTime('2020-01-07 12:12:12', $timezone));
$timesheet4 = new Timesheet();
$timesheet4->setProject($project);
$timesheet4->setBegin(new \DateTime('2020-01-27 10:12:12', $timezone));
$timesheet4->setEnd(new \DateTime('2020-11-27 11:12:12', $timezone));
$timesheet5 = new Timesheet();
$timesheet5->setProject($project);
$timesheet5->setBegin(new \DateTime('2012-01-27 12:12:12', $timezone));
$timesheet5->setEnd(new \DateTime('2018-01-27 12:12:12', $timezone));
$repo = $this->createMock(InvoiceItemRepositoryInterface::class);
$repo->method('getInvoiceItemsForQuery')->willReturn([
$timesheet1,
$timesheet2,
$timesheet3,
$timesheet4,
$timesheet5,
]);
$template = new InvoiceTemplate();
$template->setNumberGenerator('date');
$template->setLanguage('de');
$query = new InvoiceQuery();
$query->setCustomers([new Customer('foo'), $customer]);
$query->setTemplate($template);
self::assertNull($query->getBegin());
self::assertNull($query->getEnd());
$sut = $this->getSut([]);
$sut->addCalculator(new DefaultCalculator());
$sut->addNumberGenerator($this->getNumberGeneratorSut());
$sut->addInvoiceItemRepository($repo);
$sut->createModels($query);
self::assertNotNull($query->getBegin());
self::assertNotNull($query->getEnd());
self::assertEquals('2010-01-27T00:00:00+01:00', $query->getBegin()->format(DATE_ATOM));
self::assertEquals('2020-11-27T23:59:59+01:00', $query->getEnd()->format(DATE_ATOM));
}
public function testCreateModelsIncludesModelsWithNegativeTotal(): void
{
$timezone = new \DateTimeZone('Europe/Vienna');
$customer1 = $this->createMock(Customer::class);
$customer1->method('getId')->willReturn(1);
$customer1->method('getName')->willReturn('customer1');
$customer1->method('isVisible')->willReturn(true);
$project1 = new Project();
$project1->setCustomer($customer1);
$customer2 = $this->createMock(Customer::class);
$customer2->method('getId')->willReturn(2);
$customer2->method('getName')->willReturn('customer2');
$customer2->method('isVisible')->willReturn(true);
$project2 = new Project();
$project2->setCustomer($customer2);
$customer3 = $this->createMock(Customer::class);
$customer3->method('getId')->willReturn(3);
$customer3->method('getName')->willReturn('customer3');
$customer3->method('isVisible')->willReturn(true);
$project3 = new Project();
$project3->setCustomer($customer3);
$customer4 = $this->createMock(Customer::class);
$customer4->method('getId')->willReturn(4);
$customer4->method('getName')->willReturn('customer4');
$customer4->method('isVisible')->willReturn(true);
$project4 = new Project();
$project4->setCustomer($customer4);
$timesheet1 = new Timesheet();
$timesheet1->setProject($project1);
$timesheet1->setBegin(new \DateTime('2011-01-27 11:11:11', $timezone));
$timesheet1->setEnd(new \DateTime('2020-01-27 12:12:12', $timezone));
$timesheet1->setRate(-20.01);
$timesheet2 = new Timesheet();
$timesheet2->setProject($project1);
$timesheet2->setBegin(new \DateTime('2010-01-28 11:11:11', $timezone));
$timesheet2->setEnd(new \DateTime('2019-01-28 12:12:12', $timezone));
$timesheet2->setRate(20.0);
$timesheet3 = new Timesheet();
$timesheet3->setProject($project2);
$timesheet3->setBegin(new \DateTime('2019-01-27 22:22:22', $timezone));
$timesheet3->setEnd(new \DateTime('2020-01-07 23:23:23', $timezone));
$timesheet3->setRate(100);
$timesheet4 = new Timesheet();
$timesheet4->setProject($project2);
$timesheet4->setBegin(new \DateTime('2019-01-28 22:22:22', $timezone));
$timesheet4->setEnd(new \DateTime('2020-01-08 23:22:22', $timezone));
$timesheet4->setRate(-200);
$timesheet5 = new Timesheet();
$timesheet5->setProject($project3);
$timesheet5->setBegin(new \DateTime('2012-01-27 12:12:12', $timezone));
$timesheet5->setEnd(new \DateTime('2018-01-27 12:12:12', $timezone));
$timesheet5->setRate(1.73);
$timesheet6 = new Timesheet();
$timesheet6->setProject($project4);
$timesheet6->setBegin(new \DateTime('2011-01-27 11:11:11', $timezone));
$timesheet6->setEnd(new \DateTime('2020-01-27 12:12:12', $timezone));
$timesheet6->setRate(-20.0);
$timesheet7 = new Timesheet();
$timesheet7->setProject($project4);
$timesheet7->setBegin(new \DateTime('2010-01-28 11:11:11', $timezone));
$timesheet7->setEnd(new \DateTime('2019-01-28 12:12:12', $timezone));
$timesheet7->setRate(20.0);
$repo = $this->createMock(InvoiceItemRepositoryInterface::class);
$repo->method('getInvoiceItemsForQuery')->willReturn([
$timesheet1,
$timesheet2,
$timesheet3,
$timesheet4,
$timesheet5,
$timesheet6,
$timesheet7,
]);
$template = new InvoiceTemplate();
$template->setNumberGenerator('date');
$template->setLanguage('de');
self::assertEquals('de', $template->getLanguage());
$query = new InvoiceQuery();
$query->setCustomers([$customer3, new Customer('tröööt'), $customer1, $customer3]);
$query->setTemplate($template);
self::assertNull($query->getBegin());
self::assertNull($query->getEnd());
$sut = $this->getSut([]);
$sut->addCalculator(new DefaultCalculator());
$sut->addNumberGenerator($this->getNumberGeneratorSut());
$sut->addInvoiceItemRepository($repo);
$models = $sut->createModels($query);
self::assertCount(4, $models);
self::assertInstanceOf(InvoiceModel::class, $models[0]);
self::assertEquals(-0.01, $models[0]->getCalculator()->getTotal());
self::assertInstanceOf(InvoiceModel::class, $models[1]);
self::assertEquals(-100, $models[1]->getCalculator()->getTotal());
self::assertInstanceOf(InvoiceModel::class, $models[2]);
self::assertEquals(1.73, $models[2]->getCalculator()->getTotal());
self::assertInstanceOf(InvoiceModel::class, $models[3]);
self::assertEquals(0.0, $models[3]->getCalculator()->getTotal());
}
private function getNumberGeneratorSut(): DateNumberGenerator
{
$repository = $this->createMock(InvoiceRepository::class);
$repository
->expects($this->any())
->method('hasInvoice')
->willReturn(false);
return new DateNumberGenerator($repository);
}
}