Skip to content

Commit

Permalink
added support for weekly invoices (kimai#1780)
Browse files Browse the repository at this point in the history
  • Loading branch information
maarten-k authored and Sergii committed Sep 29, 2020
1 parent 1abba07 commit e0c1faa
Show file tree
Hide file tree
Showing 7 changed files with 183 additions and 0 deletions.
32 changes: 32 additions & 0 deletions src/Invoice/Calculator/WeeklyInvoiceCalculator.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
<?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\Invoice\Calculator;

use App\Invoice\CalculatorInterface;
use App\Invoice\InvoiceItemInterface;

/**
* A calculator that sums up the invoice item records per week.
*/
class WeeklyInvoiceCalculator extends AbstractSumInvoiceCalculator implements CalculatorInterface
{
protected function calculateSumIdentifier(InvoiceItemInterface $invoiceItem): string
{
return $invoiceItem->getBegin()->format('W');
}

/**
* @return string
*/
public function getId(): string
{
return 'weekly';
}
}
2 changes: 2 additions & 0 deletions src/Invoice/Hydrator/InvoiceItemDefaultHydrator.php
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,8 @@ public function hydrate(InvoiceItem $item): array
'entry.end_time' => $formatter->getFormattedTime($end),
'entry.end_timestamp' => $end->getTimestamp(),
'entry.date' => $formatter->getFormattedDateTime($begin),
'entry.week' => \intval($begin->format('W')),
'entry.weekyear' => $begin->format('o'),
'entry.user_id' => $user->getId(),
'entry.user_name' => $user->getUsername(),
'entry.user_title' => $user->getTitle(),
Expand Down
137 changes: 137 additions & 0 deletions tests/Invoice/Calculator/WeeklyInvoiceCalculatorTest.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,137 @@
<?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\Calculator;

use App\Entity\Activity;
use App\Entity\Customer;
use App\Entity\InvoiceTemplate;
use App\Entity\Project;
use App\Entity\Timesheet;
use App\Entity\User;
use App\Invoice\Calculator\WeeklyInvoiceCalculator;
use App\Invoice\InvoiceModel;
use App\Repository\Query\InvoiceQuery;
use App\Tests\Invoice\DebugFormatter;
use DateTime;

/**
* @covers \App\Invoice\Calculator\WeeklyInvoiceCalculator
* @covers \App\Invoice\Calculator\AbstractSumInvoiceCalculator
* @covers \App\Invoice\Calculator\AbstractMergedCalculator
* @covers \App\Invoice\Calculator\AbstractCalculator
*/
class WeeklyInvoiceCalculatorTest extends AbstractCalculatorTest
{
public function testEmptyModel()
{
$this->assertEmptyModel(new WeeklyInvoiceCalculator());
}

public function testWithMultipleEntries()
{
$customer = new Customer();
$template = new InvoiceTemplate();
$template->setVat(19);

$user = $this->getMockBuilder(User::class)->onlyMethods(['getId'])->disableOriginalConstructor()->getMock();
$user->method('getId')->willReturn(1);

$project1 = $this->getMockBuilder(Project::class)->onlyMethods(['getId'])->disableOriginalConstructor()->getMock();
$project1->method('getId')->willReturn(1);

$project2 = $this->getMockBuilder(Project::class)->onlyMethods(['getId'])->disableOriginalConstructor()->getMock();
$project2->method('getId')->willReturn(2);

$project3 = $this->getMockBuilder(Project::class)->onlyMethods(['getId'])->disableOriginalConstructor()->getMock();
$project3->method('getId')->willReturn(3);

$timesheet = new Timesheet();
$timesheet
->setBegin(new DateTime('2018-11-29'))
->setEnd(new DateTime())
->setDuration(3600)
->setRate(293.27)
->setUser($user)
->setActivity((new Activity())->setName('sdsd'))
->setProject($project1);

$timesheet2 = new Timesheet();
$timesheet2
->setBegin(new DateTime('2018-11-29'))
->setEnd(new DateTime())
->setDuration(400)
->setRate(84.75)
->setUser($user)
->setActivity((new Activity())->setName('bar'))
->setProject($project2);

$timesheet3 = new Timesheet();
$timesheet3
->setBegin(new DateTime('2018-11-28'))
->setEnd(new DateTime())
->setDuration(1800)
->setRate(111.11)
->setUser($user)
->setActivity((new Activity())->setName('foo'))
->setProject($project1);

$timesheet4 = new Timesheet();
$timesheet4
->setBegin(new DateTime())
->setEnd(new DateTime('2018-11-28'))
->setDuration(400)
->setRate(1947.99)
->setUser($user)
->setActivity((new Activity())->setName('blub'))
->setProject($project2);

$timesheet5 = new Timesheet();
$timesheet5
->setBegin(new DateTime('2018-11-28'))
->setEnd(new DateTime())
->setDuration(400)
->setRate(84)
->setUser(new User())
->setActivity(new Activity())
->setProject($project3);

$entries = [$timesheet, $timesheet2, $timesheet3, $timesheet4, $timesheet5];

$query = new InvoiceQuery();
$query->setProjects([$project1]);

$model = new InvoiceModel(new DebugFormatter());
$model->setCustomer($customer);
$model->setTemplate($template);
$model->addEntries($entries);
$model->setQuery($query);

$sut = new WeeklyInvoiceCalculator();
$sut->setModel($model);

$this->assertEquals('weekly', $sut->getId());
$this->assertEquals(3000.13, $sut->getTotal());
$this->assertEquals(19, $sut->getVat());
$this->assertEquals('EUR', $model->getCurrency());
$this->assertEquals(2521.12, $sut->getSubtotal());
$this->assertEquals(6600, $sut->getTimeWorked());

$entries = $sut->getEntries();
self::assertCount(2, $entries);
$this->assertEquals(573.13, $entries[0]->getRate());
$this->assertEquals(1947.99, $entries[1]->getRate());
self::assertEquals(2521.12, $entries[0]->getRate() + $entries[1]->getRate());
}

public function testDescriptionByTimesheet()
{
$this->assertDescription(new WeeklyInvoiceCalculator(), false, false);
}
}
2 changes: 2 additions & 0 deletions tests/Invoice/Hydrator/InvoiceItemDefaultHydratorTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,8 @@ protected function assertEntryStructure(array $model, array $metaFields)
'entry.end_time',
'entry.end_timestamp',
'entry.date',
'entry.week',
'entry.weekyear',
'entry.user_id',
'entry.user_name',
'entry.user_alias',
Expand Down
2 changes: 2 additions & 0 deletions tests/Invoice/Renderer/DebugRendererTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -225,6 +225,8 @@ protected function assertEntryStructure(array $model, array $metaFields)
'entry.end_time',
'entry.end_timestamp',
'entry.date',
'entry.week',
'entry.weekyear',
'entry.user_id',
'entry.user_name',
'entry.user_alias',
Expand Down
4 changes: 4 additions & 0 deletions translations/invoice-calculator.de.xlf
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,10 @@
<source>date</source>
<target>Datum: ein Eintrag pro Tag (verwendet Startdatum)</target>
</trans-unit>
<trans-unit id="weekly">
<source>weekly</source>
<target>Wöchentlich: ein Eintrag pro Woche (verwendet Startdatum)</target>
</trans-unit>
</body>
</file>
</xliff>
4 changes: 4 additions & 0 deletions translations/invoice-calculator.en.xlf
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,10 @@
<source>date</source>
<target>Date: one entry per day (uses start date)</target>
</trans-unit>
<trans-unit id="weekly">
<source>weekly</source>
<target>Weekly: one entry per week (uses start date)</target>
</trans-unit>
</body>
</file>
</xliff>

0 comments on commit e0c1faa

Please sign in to comment.