diff --git a/.github/workflows/code-analysis.yaml b/.github/workflows/code-analysis.yaml new file mode 100644 index 0000000..a07f0f4 --- /dev/null +++ b/.github/workflows/code-analysis.yaml @@ -0,0 +1,37 @@ +name: Code Analysis + +on: + pull_request: + workflow_dispatch: +permissions: + contents: read +jobs: + code_analysis: + strategy: + fail-fast: false + matrix: + actions: + - name: 'PHPStan' + run: composer phpstan + - name: 'Coding Standards' + run: composer check-cs + name: ${{ matrix.actions.name }} + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - name: Cache Composer dependencies + uses: actions/cache@v4 + with: + path: /tmp/composer-cache + key: ${{ runner.os }}-${{ hashFiles('**/composer.lock') }} + - name: Setup PHP + id: setup-php + uses: shivammathur/setup-php@v2 + with: + php-version: 8.2 + extensions: 'ctype,curl,dom,iconv,imagick,intl,json,mbstring,openssl,pcre,pdo,reflection,spl,zip' + ini-values: post_max_size=256M, max_execution_time=180, memory_limit=512M + tools: composer:v2 + - name: Install Composer dependencies + run: composer install --no-interaction --no-ansi --no-progress + - run: ${{ matrix.actions.run }} \ No newline at end of file diff --git a/CHANGELOG.md b/CHANGELOG.md index f03f2f7..aa610bf 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,6 +4,11 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](http://keepachangelog.com/) and this project adheres to [Semantic Versioning](http://semver.org/). +## 5.0.0 - 2024-04-23 + +### Added +- Craft 5 support + ## 3.0.0 - 2022-10-03 ### Added diff --git a/README.md b/README.md index 3ddebd0..8f70846 100644 --- a/README.md +++ b/README.md @@ -75,7 +75,7 @@ To set the header of the file (the first row): To set the filename: ```twig {% set currentDate = now|date('Y-m-d') %} -{% do beam.setFilename('report-#{currentDate}') %} +{% do beam.setFilename("report-#{currentDate}") %} ``` To overwrite the content: diff --git a/composer.json b/composer.json index 29a2709..1cd4032 100644 --- a/composer.json +++ b/composer.json @@ -2,7 +2,7 @@ "name": "superbig/craft3-beam", "description": "Generate CSVs and XLS files in your templates", "type": "craft-plugin", - "version": "3.0.0", + "version": "5.0.0", "keywords": [ "craft", "cms", @@ -22,9 +22,14 @@ } ], "require": { - "craftcms/cms": "^4.0.0", - "league/csv": "^8.1|^9.0", - "mk-j/php_xlsxwriter": "^0.38.0" + "craftcms/cms": "^5.0.0", + "league/csv": "^9.0", + "mk-j/php_xlsxwriter": "^0.39.0" + }, + "require-dev": { + "craftcms/ecs": "dev-main", + "craftcms/phpstan": "dev-main", + "craftcms/rector": "dev-main" }, "repositories": [ { @@ -37,6 +42,19 @@ "superbig\\beam\\": "src/" } }, + "scripts": { + "phpstan": "phpstan --ansi --memory-limit=1G", + "check-cs": "ecs check --ansi", + "fix-cs": "ecs check --fix --ansi" + }, + "config": { + "allow-plugins": { + "craftcms/plugin-installer": true, + "yiisoft/yii2-composer": true + }, + "optimize-autoloader": true, + "sort-packages": true + }, "extra": { "name": "Beam", "handle": "beam", diff --git a/ecs.php b/ecs.php new file mode 100644 index 0000000..b5fe0ae --- /dev/null +++ b/ecs.php @@ -0,0 +1,14 @@ +paths([ + __DIR__ . '/src', + __FILE__, + ]); + + $ecsConfig->parallel(); + $ecsConfig->sets([SetList::CRAFT_CMS_4]); +}; diff --git a/phpstan.neon b/phpstan.neon new file mode 100644 index 0000000..8bc0882 --- /dev/null +++ b/phpstan.neon @@ -0,0 +1,6 @@ +includes: + - %currentWorkingDirectory%/vendor/craftcms/phpstan/phpstan.neon +parameters: + level: 5 + paths: + - src \ No newline at end of file diff --git a/src/Beam.php b/src/Beam.php index 2b17ca6..5d9b93d 100644 --- a/src/Beam.php +++ b/src/Beam.php @@ -10,16 +10,14 @@ namespace superbig\beam; -use superbig\beam\services\BeamService as BeamServiceService; -use superbig\beam\variables\BeamVariable; - use Craft; use craft\base\Plugin; -use craft\services\Plugins; -use craft\events\PluginEvent; -use craft\web\UrlManager; -use craft\web\twig\variables\CraftVariable; + use craft\events\RegisterUrlRulesEvent; +use craft\web\twig\variables\CraftVariable; +use craft\web\UrlManager; +use superbig\beam\services\BeamService as BeamServiceService; +use superbig\beam\variables\BeamVariable; use yii\base\Event; @@ -56,7 +54,7 @@ public function init() Event::on( UrlManager::class, UrlManager::EVENT_REGISTER_SITE_URL_RULES, - function (RegisterUrlRulesEvent $event) { + function(RegisterUrlRulesEvent $event) { $event->rules['beam/download'] = 'beam/default'; } ); @@ -64,7 +62,7 @@ function (RegisterUrlRulesEvent $event) { Event::on( UrlManager::class, UrlManager::EVENT_REGISTER_CP_URL_RULES, - function (RegisterUrlRulesEvent $event) { + function(RegisterUrlRulesEvent $event) { $event->rules['beam/download'] = 'beam/default'; } ); @@ -72,7 +70,7 @@ function (RegisterUrlRulesEvent $event) { Event::on( CraftVariable::class, CraftVariable::EVENT_INIT, - function (Event $event) { + function(Event $event) { /** @var CraftVariable $variable */ $variable = $event->sender; $variable->set('beam', BeamVariable::class); diff --git a/src/controllers/DefaultController.php b/src/controllers/DefaultController.php index bc18472..4d1adb7 100644 --- a/src/controllers/DefaultController.php +++ b/src/controllers/DefaultController.php @@ -10,10 +10,10 @@ namespace superbig\beam\controllers; -use superbig\beam\Beam; - use Craft; + use craft\web\Controller; +use superbig\beam\Beam; use yii\web\NotFoundHttpException; /** @@ -23,20 +23,8 @@ */ class DefaultController extends Controller { - - // Protected Properties - // ========================================================================= - - /** - * @var bool|array Allows anonymous access to this controller's actions. - * The actions must be in 'kebab-case' - * @access protected - */ protected array|int|bool $allowAnonymous = ['index']; - // Public Methods - // ========================================================================= - /** * @return mixed * @throws NotFoundHttpException @@ -45,7 +33,7 @@ class DefaultController extends Controller public function actionIndex() { $request = Craft::$app->getRequest(); - $hash = $request->getRequiredParam('hash'); + $hash = $request->getRequiredParam('hash'); $config = Beam::$plugin->beamService->downloadHash($hash); @@ -53,7 +41,7 @@ public function actionIndex() throw new NotFoundHttpException(); } - $path = $config['path']; + $path = $config['path']; $filename = $config['filename']; return Craft::$app->getResponse()->sendFile($path, $filename, [ diff --git a/src/models/BeamModel.php b/src/models/BeamModel.php index db7aa80..3646c67 100644 --- a/src/models/BeamModel.php +++ b/src/models/BeamModel.php @@ -10,11 +10,10 @@ namespace superbig\beam\models; -use superbig\beam\Beam; - -use Craft; use craft\base\Model; +use superbig\beam\Beam; + /** * @author Superbig * @package Beam @@ -22,26 +21,11 @@ */ class BeamModel extends Model { - // Public Properties - // ========================================================================= - - /** @var array */ - public $header = []; - - /** @var array */ - public $content = []; - - /** @var array */ - public $rows = []; - - /** @var string */ - public $filename = 'output'; - - /** @var string */ - public $sheetName = 'Sheet'; - - // Public Methods - // ========================================================================= + public array $header = []; + public array $content = []; + public array $rows = []; + public string $filename = 'output'; + public string $sheetName = 'Sheet'; public function init(): void { @@ -52,12 +36,7 @@ public function init(): void } } - /** - * @param array $content - * - * @return $this - */ - public function append(array $content = []) + public function append(array $content = []): static { if (isset($content[0]) && !\is_array($content[0])) { $content = [$content]; @@ -68,35 +47,30 @@ public function append(array $content = []) return $this; } - public function setHeader($headers = []) + public function setHeader(array $headers = []): static { $this->header = $headers; return $this; } - public function getContent() + public function getContent(): array { return $this->content; } - /** - * @param $content - * - * @return $this - */ - public function setContent($content) + public function setContent(array $content): static { $this->content = $content; return $this; } - public function getConfig() + public function getConfig(): array { return [ 'header' => $this->header, - 'rows' => $this->content, + 'rows' => $this->content, ]; } @@ -107,39 +81,31 @@ public function getFilename($ext = null) return "$filename.$ext"; } - public function setFilename($filename = null) + public function setFilename($filename = null): static { $this->filename = $filename; return $this; } - public function csv($filename = null) + public function csv($filename = null): void { if ($filename) { $this->filename = $filename; } - return Beam::$plugin->beamService->csv($this); + Beam::$plugin->beamService->csv($this); } - public function xlsx($filename = null) + public function xlsx($filename = null): void { if ($filename) { $this->filename = $filename; } - return Beam::$plugin->beamService->xlsx($this); - } - - public function html() - { - + Beam::$plugin->beamService->xlsx($this); } - /** - * @inheritdoc - */ public function rules(): array { return [ diff --git a/src/services/BeamService.php b/src/services/BeamService.php index c1a978e..43bb66e 100644 --- a/src/services/BeamService.php +++ b/src/services/BeamService.php @@ -10,19 +10,21 @@ namespace superbig\beam\services; +use Craft; +use craft\base\Component; use craft\helpers\FileHelper; -use craft\helpers\Path; use craft\helpers\StringHelper; -use craft\helpers\UrlHelper; -use superbig\beam\Beam; -use Craft; -use craft\base\Component; +use craft\helpers\UrlHelper; use League\Csv\Writer; -use League\Csv\Reader; +use superbig\beam\Beam; use superbig\beam\models\BeamModel; use XLSXWriter; -use yii\web\Response; +use yii\base\ErrorException; +use yii\base\Exception; +use yii\base\ExitException; +use yii\base\InvalidConfigException; +use yii\base\InvalidRouteException; /** * @author Superbig @@ -31,9 +33,6 @@ */ class BeamService extends Component { - // Public Methods - // ========================================================================= - public function create($config = []) { $model = new BeamModel($config); @@ -44,16 +43,16 @@ public function create($config = []) /** * @param BeamModel $model * - * @return null + * @return void * @throws \League\Csv\CannotInsertRecord */ - public function csv(BeamModel $model) + public function csv(BeamModel $model): void { - $header = $model->header; + $header = $model->header; $content = $model->content; if (empty($header) && empty($content)) { - return null; + return; } $csv = Writer::createFromString(''); @@ -76,19 +75,20 @@ public function csv(BeamModel $model) } /** - * @param BeamModel $model - * - * @return null - * @throws \yii\base\Exception + * @throws ErrorException + * @throws Exception + * @throws ExitException + * @throws InvalidConfigException + * @throws InvalidRouteException */ - public function xlsx(BeamModel $model) + public function xlsx(BeamModel $model): void { $tempPath = Craft::$app->path->getTempPath() . DIRECTORY_SEPARATOR . 'beam' . DIRECTORY_SEPARATOR; - $header = $model->header; - $content = $model->content; + $header = $model->header; + $content = $model->content; if (empty($header) && empty($content)) { - return null; + return; } if (!file_exists($tempPath) && !is_dir($tempPath)) { @@ -96,7 +96,7 @@ public function xlsx(BeamModel $model) } // Load the CSV document from a string - $writer = new XLSXWriter(); + $writer = new XLSXWriter(); $sheetName = !empty($model->sheetName) ? $model->sheetName : 'Sheet'; if (!empty($header)) { @@ -122,13 +122,20 @@ public function xlsx(BeamModel $model) $this->writeAndRedirect($writer->writeToString(), $model->getFilename('xlsx'), $mimeType); } - public function downloadHash($fileHash = null) + /** + * @param $fileHash + * @return array|bool + * @throws Exception + * @throws InvalidConfigException + */ + public function downloadHash($fileHash = null): array | bool { $hash = Craft::$app->getSecurity()->validateData($fileHash); if (!$hash) { return false; } + $config = $this->unhashConfig($hash); $config['path'] = Craft::$app->path->getTempPath() . DIRECTORY_SEPARATOR . 'beam' . DIRECTORY_SEPARATOR . $config['tempFilename']; @@ -136,19 +143,26 @@ public function downloadHash($fileHash = null) return $config; } - private function writeAndRedirect($content, $filename, $mimeType) + /** + * @throws InvalidRouteException + * @throws InvalidConfigException + * @throws ErrorException + * @throws Exception + * @throws ExitException + */ + private function writeAndRedirect(string $content, string $filename, string $mimeType): void { - $tempPath = Craft::$app->path->getTempPath() . DIRECTORY_SEPARATOR . 'beam' . DIRECTORY_SEPARATOR; + $tempPath = Craft::$app->path->getTempPath() . DIRECTORY_SEPARATOR . 'beam' . DIRECTORY_SEPARATOR; $tempFilename = StringHelper::randomString(12) . "-{$filename}"; - $config = [ - 'filename' => $filename, + $config = [ + 'filename' => $filename, 'tempFilename' => $tempFilename, - 'mimeType' => $mimeType, + 'mimeType' => $mimeType, ]; $hashConfig = $this->hashConfig($config); $verifyHash = Craft::$app->getSecurity()->hashData($hashConfig); - $url = UrlHelper::siteUrl('beam/download', [ + $url = UrlHelper::siteUrl('beam/download', [ 'hash' => $verifyHash, ]); @@ -156,7 +170,7 @@ private function writeAndRedirect($content, $filename, $mimeType) Craft::$app->getResponse()->redirect($url); - return Craft::$app->end(); + Craft::$app->end(); } public function hashConfig($config = []): string @@ -171,18 +185,19 @@ public function unhashConfig(string $hash): array $config = base64_decode($hash); $config = explode('||', $config); - list ($filename, $tempFilename, $mimeType) = $config; + list($filename, $tempFilename, $mimeType) = $config; $config = [ - 'filename' => $filename, + 'filename' => $filename, 'tempFilename' => $tempFilename, - 'mimeType' => $mimeType, + 'mimeType' => $mimeType, ]; return $config; } - private function normalizeCellFormat(string $type) { + private function normalizeCellFormat(string $type): string + { $types = [ 'number' => 'integer', 'date' => 'date', diff --git a/src/variables/BeamVariable.php b/src/variables/BeamVariable.php index f42e137..cfb2189 100644 --- a/src/variables/BeamVariable.php +++ b/src/variables/BeamVariable.php @@ -12,8 +12,6 @@ use superbig\beam\Beam; -use Craft; - /** * @author Superbig * @package Beam @@ -29,7 +27,7 @@ class BeamVariable * * @return null */ - public function create ($options = []) + public function create($options = []) { return Beam::$plugin->beamService->create($options); }