Skip to content

Commit

Permalink
improve export filename (#2958)
Browse files Browse the repository at this point in the history
  • Loading branch information
kevinpapst committed Nov 19, 2021
1 parent 56d0267 commit 8591313
Show file tree
Hide file tree
Showing 10 changed files with 199 additions and 10 deletions.
6 changes: 4 additions & 2 deletions src/Export/Base/AbstractSpreadsheetRenderer.php
Expand Up @@ -16,6 +16,7 @@
use App\Event\ProjectMetaDisplayEvent;
use App\Event\TimesheetMetaDisplayEvent;
use App\Event\UserPreferenceDisplayEvent;
use App\Export\ExportFilename;
use App\Export\ExportItemInterface;
use App\Repository\Query\CustomerQuery;
use App\Repository\Query\TimesheetQuery;
Expand Down Expand Up @@ -753,9 +754,10 @@ protected function isTotalRowSupported(): bool
public function render(array $exportItems, TimesheetQuery $query): Response
{
$spreadsheet = $this->fromArrayToSpreadsheet($exportItems, $query);
$filename = $this->saveSpreadsheet($spreadsheet);
$file = $this->saveSpreadsheet($spreadsheet);
$filename = new ExportFilename($query);

return $this->getFileResponse($filename, 'kimai-export' . $this->getFileExtension());
return $this->getFileResponse($file, $filename->getFilename() . $this->getFileExtension());
}

/**
Expand Down
7 changes: 5 additions & 2 deletions src/Export/Base/PDFRenderer.php
Expand Up @@ -10,6 +10,7 @@
namespace App\Export\Base;

use App\Export\ExportContext;
use App\Export\ExportFilename;
use App\Export\ExportItemInterface;
use App\Project\ProjectStatisticService;
use App\Repository\Query\TimesheetQuery;
Expand Down Expand Up @@ -94,8 +95,9 @@ public function setPdfOption(string $key, string $value): PDFRenderer
*/
public function render(array $timesheets, TimesheetQuery $query): Response
{
$filename = new ExportFilename($query);
$context = new ExportContext();
$context->setOption('filename', 'kimai-export');
$context->setOption('filename', $filename->getFilename());

$summary = $this->calculateSummary($timesheets);
$content = $this->twig->render($this->getTemplate(), array_merge([
Expand All @@ -117,7 +119,8 @@ public function render(array $timesheets, TimesheetQuery $query): Response

$filename = $context->getOption('filename');
if (empty($filename)) {
$filename = 'kimai-export';
$filename = new ExportFilename($query);
$filename = $filename->getFilename();
}

$filename = FileHelper::convertToAsciiFilename($filename);
Expand Down
82 changes: 82 additions & 0 deletions src/Export/ExportFilename.php
@@ -0,0 +1,82 @@
<?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\Export;

use App\Entity\Customer;
use App\Repository\Query\TimesheetQuery;
use App\Utils\FileHelper;

final class ExportFilename
{
/**
* @var string
*/
private $filename;

public function __construct(TimesheetQuery $query)
{
$filename = date('Ymd');
$hasName = false;

$customers = $query->getCustomers();
if (\count($customers) === 1) {
$filename .= '-' . $this->convert($this->getCustomerName($customers[0]));
$hasName = true;
}

$projects = $query->getProjects();
if (\count($projects) === 1) {
if (!$hasName) {
$filename .= '-' . $this->convert($this->getCustomerName($projects[0]->getCustomer()));
}
$filename .= '-' . $this->convert($projects[0]->getName());
$hasName = true;
}

$users = $query->getUsers();
if (\count($users) === 1) {
$filename .= '-' . $this->convert($users[0]->getDisplayName());
$hasName = true;
}

if (!$hasName) {
$filename .= '-kimai-export';
}

$filename = str_replace(['/', '\\'], '-', $filename);

$this->filename = $filename;
}

private function getCustomerName(Customer $customer): string
{
$company = $customer->getCompany();
if (empty($company)) {
$company = $customer->getName();
}

return $company;
}

private function convert(string $filename): string
{
return FileHelper::convertToAsciiFilename($filename);
}

public function getFilename()
{
return $this->filename;
}

public function __toString()
{
return $this->getFilename();
}
}
96 changes: 96 additions & 0 deletions tests/Export/ExportFilenameTest.php
@@ -0,0 +1,96 @@
<?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\Export;

use App\Entity\Customer;
use App\Entity\Project;
use App\Entity\User;
use App\Export\ExportFilename;
use App\Repository\Query\TimesheetQuery;
use PHPUnit\Framework\TestCase;

/**
* @covers \App\Export\ExportFilename
*/
class ExportFilenameTest extends TestCase
{
public function testExportFilename()
{
$datePrefix = date('Ymd');

$query = new TimesheetQuery();

$sut = new ExportFilename($query);

self::assertEquals($datePrefix . '-kimai-export', $sut->getFilename());
self::assertEquals($datePrefix . '-kimai-export', (string) $sut);

$customer = new Customer();
$customer->setName('foo');
$query = new TimesheetQuery();
$query->addCustomer($customer);
$sut = new ExportFilename($query);

self::assertEquals($datePrefix . '-foo', $sut->getFilename());
self::assertEquals($datePrefix . '-foo', (string) $sut);

$customer->setCompany('barß / laölala # ldksjf 123 MyAwesome GmbH');
$sut = new ExportFilename($query);

self::assertEquals($datePrefix . '-barss_laolala_ldksjf_123_MyAwesome_GmbH', $sut->getFilename());
self::assertEquals($datePrefix . '-barss_laolala_ldksjf_123_MyAwesome_GmbH', (string) $sut);

$customer->setCompany('까깨꺄꺠꺼께껴꼐꼬꽈sssss');
$sut = new ExportFilename($query);
self::assertEquals($datePrefix . '-kkakkaekkyakkyaekkeokkekkyeokkyekkokkwasssss', $sut->getFilename());

$customer->setCompany('\"#+ß.!$%&/()=?\\n=/*-+´_<>@' . "\n");
$sut = new ExportFilename($query);
self::assertEquals($datePrefix . '-ss_n_-', $sut->getFilename());

$project = new Project();
$project->setName('Demo ProjecT1');
$customer->setCompany('\"#+ß.!$%&/()=?\\n=/*-+´_<>@' . "\n");
$query->addProject($project);

$sut = new ExportFilename($query);
self::assertEquals($datePrefix . '-ss_n_--Demo_ProjecT1', $sut->getFilename());

$user = new User();
$user->setUsername('thorsten');
$query->addUser($user);

$sut = new ExportFilename($query);
self::assertEquals($datePrefix . '-ss_n_--Demo_ProjecT1-thorsten', $sut->getFilename());
$user->setAlias('Martin Müller-Lüdenscheidt');

$sut = new ExportFilename($query);
self::assertEquals($datePrefix . '-ss_n_--Demo_ProjecT1-Martin_Muller-Ludenscheidt', $sut->getFilename());

$user = $this->createMock(User::class);
$user->method('getId')->willReturn(1);
$query->addUser($user);
$sut = new ExportFilename($query);
self::assertEquals($datePrefix . '-ss_n_--Demo_ProjecT1', $sut->getFilename());

$project = new Project();
$project->setName('Project2');
$query->addProject($project);
$sut = new ExportFilename($query);
self::assertEquals($datePrefix . '-ss_n_-', $sut->getFilename());

$customer = new Customer();
$customer->setName('Customer AAAA');
$query->addCustomer($customer);

$sut = new ExportFilename($query);
self::assertEquals($datePrefix . '-kimai-export', $sut->getFilename());
}
}
3 changes: 2 additions & 1 deletion tests/Export/Renderer/CsvRendererTest.php
Expand Up @@ -49,8 +49,9 @@ public function testRender($totalDuration, $totalRate, $expectedRate, $expectedR
$response = $this->render($sut);

$file = $response->getFile();
$prefix = date('Ymd');
$this->assertEquals('text/csv', $response->headers->get('Content-Type'));
$this->assertEquals('attachment; filename=kimai-export.csv', $response->headers->get('Content-Disposition'));
$this->assertEquals('attachment; filename=' . $prefix . '-Customer_Name-project_name.csv', $response->headers->get('Content-Disposition'));

$this->assertTrue(file_exists($file->getRealPath()));
$content = file_get_contents($file->getRealPath());
Expand Down
3 changes: 2 additions & 1 deletion tests/Export/Renderer/PdfRendererTest.php
Expand Up @@ -58,8 +58,9 @@ public function testRender()

$response = $this->render($sut);

$prefix = date('Ymd');
$this->assertEquals('application/pdf', $response->headers->get('Content-Type'));
$this->assertEquals('attachment; filename=kimai-export.pdf', $response->headers->get('Content-Disposition'));
$this->assertEquals('attachment; filename=' . $prefix . '-Customer_Name-project_name.pdf', $response->headers->get('Content-Disposition'));

$this->assertNotEmpty($response->getContent());
}
Expand Down
3 changes: 2 additions & 1 deletion tests/Export/Renderer/XlsxRendererTest.php
Expand Up @@ -39,8 +39,9 @@ public function testRender()
$response = $this->render($sut);

$file = $response->getFile();
$prefix = date('Ymd');
$this->assertEquals('application/vnd.openxmlformats-officedocument.spreadsheetml.sheet', $response->headers->get('Content-Type'));
$this->assertEquals('attachment; filename=kimai-export.xlsx', $response->headers->get('Content-Disposition'));
$this->assertEquals('attachment; filename=' . $prefix . '-Customer_Name-project_name.xlsx', $response->headers->get('Content-Disposition'));

$this->assertTrue(file_exists($file->getRealPath()));

Expand Down
3 changes: 2 additions & 1 deletion tests/Export/Timesheet/CsvRendererTest.php
Expand Up @@ -46,8 +46,9 @@ public function testRender($totalDuration, $totalRate, $expectedRate, $expectedR
$response = $this->render($sut);

$file = $response->getFile();
$prefix = date('Ymd');
$this->assertEquals('text/csv', $response->headers->get('Content-Type'));
$this->assertEquals('attachment; filename=kimai-export.csv', $response->headers->get('Content-Disposition'));
$this->assertEquals('attachment; filename=' . $prefix . '-Customer_Name-project_name.csv', $response->headers->get('Content-Disposition'));

$this->assertTrue(file_exists($file->getRealPath()));
$content = file_get_contents($file->getRealPath());
Expand Down
3 changes: 2 additions & 1 deletion tests/Export/Timesheet/PdfRendererTest.php
Expand Up @@ -51,8 +51,9 @@ public function testRender()

$response = $this->render($sut);

$prefix = date('Ymd');
$this->assertEquals('application/pdf', $response->headers->get('Content-Type'));
$this->assertEquals('attachment; filename=kimai-export.pdf', $response->headers->get('Content-Disposition'));
$this->assertEquals('attachment; filename=' . $prefix . '-Customer_Name-project_name.pdf', $response->headers->get('Content-Disposition'));

$this->assertNotEmpty($response->getContent());
}
Expand Down
3 changes: 2 additions & 1 deletion tests/Export/Timesheet/XlsxRendererTest.php
Expand Up @@ -36,8 +36,9 @@ public function testRender()
$response = $this->render($sut);

$file = $response->getFile();
$prefix = date('Ymd');
$this->assertEquals('application/vnd.openxmlformats-officedocument.spreadsheetml.sheet', $response->headers->get('Content-Type'));
$this->assertEquals('attachment; filename=kimai-export.xlsx', $response->headers->get('Content-Disposition'));
$this->assertEquals('attachment; filename=' . $prefix . '-Customer_Name-project_name.xlsx', $response->headers->get('Content-Disposition'));

$this->assertTrue(file_exists($file->getRealPath()));

Expand Down

0 comments on commit 8591313

Please sign in to comment.