mautic / app /bundles /CoreBundle /Helper /ExportHelper.php
chrisbryan17's picture
Upload folder using huggingface_hub
d2897cd verified
raw
history blame contribute delete
7.11 kB
<?php
declare(strict_types=1);
namespace Mautic\CoreBundle\Helper;
use Mautic\CoreBundle\Exception\FilePathException;
use Mautic\CoreBundle\Model\IteratorExportDataModel;
use Mautic\LeadBundle\Entity\Lead;
use PhpOffice\PhpSpreadsheet\IOFactory;
use PhpOffice\PhpSpreadsheet\Spreadsheet;
use PhpOffice\PhpSpreadsheet\Writer\Csv;
use Symfony\Component\HttpFoundation\StreamedResponse;
use Symfony\Contracts\Translation\TranslatorInterface;
/**
* Provides several functions for export-related tasks,
* like exporting to CSV or Excel.
*/
class ExportHelper
{
public const EXPORT_TYPE_EXCEL = 'xlsx';
public const EXPORT_TYPE_CSV = 'csv';
public function __construct(
private TranslatorInterface $translator,
private CoreParametersHelper $coreParametersHelper,
private FilePathResolver $filePathResolver
) {
}
/**
* Returns supported export types as an array.
*/
public function getSupportedExportTypes(): array
{
return [
self::EXPORT_TYPE_CSV,
self::EXPORT_TYPE_EXCEL,
];
}
/**
* Exports data as the given export type. You can get available export types with getSupportedExportTypes().
*
* @param array|\Iterator $data
*/
public function exportDataAs($data, string $type, string $filename): StreamedResponse
{
if (is_array($data)) {
$data = new \ArrayIterator($data);
}
if (!$data->valid()) {
throw new \Exception('No or invalid data given');
}
if (self::EXPORT_TYPE_EXCEL === $type) {
return $this->exportAsExcel($data, $filename);
}
if (self::EXPORT_TYPE_CSV === $type) {
return $this->exportAsCsv($data, $filename);
}
throw new \InvalidArgumentException($this->translator->trans('mautic.error.invalid.specific.export.type', ['%type%' => $type, '%expected_type%' => self::EXPORT_TYPE_EXCEL]));
}
public function exportDataIntoFile(IteratorExportDataModel $data, string $type, string $fileName): string
{
if (!$data->valid()) {
throw new \Exception('No or invalid data given');
}
if (self::EXPORT_TYPE_CSV === $type) {
return $this->exportAsCsvIntoFile($data, $fileName);
}
throw new \InvalidArgumentException($this->translator->trans('mautic.error.invalid.specific.export.type', ['%type%' => $type, '%expected_type%' => self::EXPORT_TYPE_CSV]));
}
public function zipFile(string $filePath, string $fileName): string
{
$zipFilePath = str_replace('.csv', '.zip', $filePath);
$zipArchive = new \ZipArchive();
if (true === $zipArchive->open($zipFilePath, \ZipArchive::OVERWRITE | \ZipArchive::CREATE)) {
$zipArchive->addFile($filePath, $fileName);
$zipArchive->close();
$this->filePathResolver->delete($filePath);
return $zipFilePath;
}
throw new FilePathException("Could not create zip archive at $zipFilePath.");
}
private function exportAsExcel(\Iterator $data, string $filename): StreamedResponse
{
$spreadsheet = $this->getSpreadsheetGeneric($data, $filename);
$objWriter = IOFactory::createWriter($spreadsheet, 'Xlsx');
$objWriter->setPreCalculateFormulas(false);
$response = new StreamedResponse(
function () use ($objWriter): void {
$objWriter->save('php://output');
}
);
$response->headers->set('Content-Type', 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet');
$response->headers->set('Content-Disposition', 'attachment; filename="'.$filename.'"');
$response->headers->set('Expires', '0');
$response->headers->set('Cache-Control', 'must-revalidate');
$response->headers->set('Pragma', 'public');
return $response;
}
private function getSpreadsheetGeneric(\Iterator $data, string $filename): Spreadsheet
{
$spreadsheet = new Spreadsheet();
$spreadsheet->getProperties()->setTitle($filename);
$spreadsheet->createSheet();
$rowCount = 2;
foreach ($data as $key => $row) {
if (0 === $key) {
// Build the header row from keys in the current row.
$spreadsheet->getActiveSheet()->fromArray(array_keys($row), null, 'A1');
}
$spreadsheet->getActiveSheet()->fromArray($row, null, "A{$rowCount}");
// Increment row
++$rowCount;
}
return $spreadsheet;
}
private function exportAsCsv(\Iterator $data, string $filename): StreamedResponse
{
$spreadsheet = $this->getSpreadsheetGeneric($data, $filename);
$objWriter = new Csv($spreadsheet);
$objWriter->setPreCalculateFormulas(false);
// For UTF-8 support
$objWriter->setUseBOM(true);
$response = new StreamedResponse(
function () use ($objWriter): void {
$objWriter->save('php://output');
}
);
$response->headers->set('Content-Type', 'text/csv');
$response->headers->set('Content-Disposition', 'attachment; filename="'.$filename.'"');
$response->headers->set('Expires', '0');
$response->headers->set('Cache-Control', 'must-revalidate');
$response->headers->set('Pragma', 'public');
return $response;
}
/**
* @param \Iterator<mixed> $data
*/
private function exportAsCsvIntoFile(\Iterator $data, string $fileName): string
{
$filePath = $this->getValidContactExportFileName($fileName);
$handler = @fopen($filePath, 'ab+');
$headerSet = false;
foreach ($data as $row) {
if (!$headerSet) {
fputcsv($handler, array_keys($row));
$headerSet = true;
}
fputcsv($handler, $row);
}
fclose($handler);
return $filePath;
}
private function getValidContactExportFileName(string $fileName): string
{
$contactExportDir = $this->coreParametersHelper->get('contact_export_dir');
$this->filePathResolver->createDirectory($contactExportDir);
$filePath = $contactExportDir.'/'.$fileName;
$fileName = (string) pathinfo($filePath, PATHINFO_FILENAME);
$extension = (string) pathinfo($filePath, PATHINFO_EXTENSION);
$originalName = $fileName;
$i = 1;
while (file_exists($filePath)) {
$fileName = $originalName.'_'.$i;
$filePath = $contactExportDir.'/'.$fileName.'.'.$extension;
++$i;
}
return $filePath;
}
/**
* @return array<string, string>
*/
public function parseLeadToExport(Lead $lead): array
{
$leadExport = $lead->getProfileFields();
$stage = $lead->getStage();
$leadExport['stage'] = $stage ? $stage->getName() : null;
return $leadExport;
}
}