![]() Server : Apache System : Linux server2.corals.io 4.18.0-348.2.1.el8_5.x86_64 #1 SMP Mon Nov 15 09:17:08 EST 2021 x86_64 User : corals ( 1002) PHP Version : 7.4.33 Disable Function : exec,passthru,shell_exec,system Directory : /home/corals/mautic.corals.io/app/bundles/CoreBundle/Helper/ |
<?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; } }