From 0c4de69aad62bbfa1a9ad747f795f36c08f554d2 Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 16:07:39 +1100 Subject: [PATCH 01/13] Updated Drupal 10 -> 11 mentions. --- .vortex/docs/content/getting-started/features.mdx | 2 +- .vortex/installer/src/Traits/PromptsTrait.php | 2 +- .vortex/tests/bats/_helper.bash | 4 ++-- README.dist.md | 4 ++-- composer.json | 2 +- 5 files changed, 7 insertions(+), 7 deletions(-) diff --git a/.vortex/docs/content/getting-started/features.mdx b/.vortex/docs/content/getting-started/features.mdx index 70c6c826b..dae410389 100644 --- a/.vortex/docs/content/getting-started/features.mdx +++ b/.vortex/docs/content/getting-started/features.mdx @@ -10,7 +10,7 @@ The following list includes āœ… completed and šŸš§ upcoming features. * šŸ’§ Drupal * āœ… Based on [drupal-composer/drupal-project](https://github.com/drupal-composer/drupal-project) - * āœ… [Drupal 10](https://www.drupal.org/project/drupal) + * āœ… [Drupal 11](https://www.drupal.org/project/drupal) * āœ… Configurable webroot * āœ… [Pre-configured settings](https://github.com/drevops/vortex/blob/develop/web/sites/default/settings.php) * āœ… [Custom module scaffold](https://github.com/drevops/vortex/blob/develop/web/modules/custom/ys_core) diff --git a/.vortex/installer/src/Traits/PromptsTrait.php b/.vortex/installer/src/Traits/PromptsTrait.php index 108a80321..ccf8930c0 100644 --- a/.vortex/installer/src/Traits/PromptsTrait.php +++ b/.vortex/installer/src/Traits/PromptsTrait.php @@ -494,7 +494,7 @@ protected function discoverValueProfile(): ?string { $this->config->getDstDir() . sprintf('/%s/profiles/custom/*/*.info.yml', $webroot), ]; - $name = File::findMatchingPath($locations, 'Drupal 10 profile implementation of'); + $name = File::findMatchingPath($locations, 'Drupal 11 profile implementation of'); if (empty($name)) { return NULL; diff --git a/.vortex/tests/bats/_helper.bash b/.vortex/tests/bats/_helper.bash index 45502d67e..9462aba07 100644 --- a/.vortex/tests/bats/_helper.bash +++ b/.vortex/tests/bats/_helper.bash @@ -1021,7 +1021,7 @@ create_fixture_readme() { cat <>"${dir}"/README.md # ${name} -Drupal 10 implementation of ${name} for ${org} +Drupal 11 implementation of ${name} for ${org} [![CircleCI](https://circleci.com/gh/your_org/your_site.svg?style=shield)](https://circleci.com/gh/your_org/your_site) @@ -1043,7 +1043,7 @@ create_fixture_composerjson() { cat <>"${dir}"/composer.json { "name": "${org_machine_name}/${machine_name}", - "description": "Drupal 10 implementation of ${name} for ${org}" + "description": "Drupal 11 implementation of ${name} for ${org}" } EOT } diff --git a/README.dist.md b/README.dist.md index ba1e89408..43ba05405 100644 --- a/README.dist.md +++ b/README.dist.md @@ -1,5 +1,5 @@ # YOURSITE -Drupal 10 implementation of YOURSITE for YOURORG +Drupal 11 implementation of YOURSITE for YOURORG [//]: # (#;< CI_PROVIDER_CIRCLECI) @@ -13,7 +13,7 @@ Drupal 10 implementation of YOURSITE for YOURORG [//]: # (#;> CI_PROVIDER_GHA) -![Drupal 10](https://img.shields.io/badge/Drupal-10-blue.svg) +![Drupal 11](https://img.shields.io/badge/Drupal-10-blue.svg) [![codecov](https://codecov.io/gh/your_org/your_site/graph/badge.svg)](https://codecov.io/gh/your_org/your_site) [//]: # (#;< RENOVATEBOT) diff --git a/composer.json b/composer.json index 6c638baf0..d7d4978c3 100644 --- a/composer.json +++ b/composer.json @@ -1,6 +1,6 @@ { "name": "your_org/your_site", - "description": "Drupal 10 implementation of YOURSITE for YOURORG", + "description": "Drupal 11 implementation of YOURSITE for YOURORG", "license": "proprietary", "type": "project", "require": { From b442eaef3eabbb092e536c53f49dcf7d8245820f Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 16:13:55 +1100 Subject: [PATCH 02/13] Added `ahoy build-installer` command. --- .vortex/.ahoy.yml | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/.vortex/.ahoy.yml b/.vortex/.ahoy.yml index 255255b0d..55e1af10e 100644 --- a/.vortex/.ahoy.yml +++ b/.vortex/.ahoy.yml @@ -8,7 +8,7 @@ commands: cmd: | [ ! -d ./tests/node_modules ] && npm --prefix tests ci [ ! -d ./docs/node_modules ] && npm --prefix docs ci - [ ! -d ./installer/node_modules ] && composer --working-dir installer install + [ ! -d ./installer/vendor ] && composer --working-dir installer install docs: name: Start documentation server. @@ -28,6 +28,12 @@ commands: [ ! -d ./docs/node_modules ] && npm --prefix docs ci npm --prefix docs run build + build-installer: + name: Build documentation. + cmd: | + [ ! -d ./installer/vendor ] && composer --working-dir installer install + composer --working-dir installer build + lint: name: Lint Vortex project. cmd: | From 1f64bcc3409083330a06eaf6f7c958fee63e1d95 Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 17:11:31 +1100 Subject: [PATCH 03/13] Updated info icon and colour in the installer. --- .../installer/src/Command/InstallCommand.php | 2 +- .vortex/installer/src/Traits/PrinterTrait.php | 13 ++++++++++++- .vortex/installer/src/Traits/TuiTrait.php | 18 ++++++++++++++++-- 3 files changed, 29 insertions(+), 4 deletions(-) diff --git a/.vortex/installer/src/Command/InstallCommand.php b/.vortex/installer/src/Command/InstallCommand.php index 2c8d21f43..edfffe7f7 100644 --- a/.vortex/installer/src/Command/InstallCommand.php +++ b/.vortex/installer/src/Command/InstallCommand.php @@ -300,7 +300,7 @@ protected function collectAnswers(): void { $this->printSummary(); if ($this->config->isInstallDebug()) { - $this->printBox($this->formatValuesList($this->getAnswers(), '', 80 - 6), 'DEBUG RESOLVED ANSWERS'); + $this->printBox($this->formatValuesList($this->getAnswers(), '', $this->getTuiWidth() - 2 - 2 * 2), 'DEBUG RESOLVED ANSWERS'); } } diff --git a/.vortex/installer/src/Traits/PrinterTrait.php b/.vortex/installer/src/Traits/PrinterTrait.php index 9de6a4a25..acda65dd8 100644 --- a/.vortex/installer/src/Traits/PrinterTrait.php +++ b/.vortex/installer/src/Traits/PrinterTrait.php @@ -13,6 +13,7 @@ protected function out(string $text, ?string $color = NULL, bool $new_line = TRU $styles = [ 'success' => "\033[0;32m%s\033[0m", 'error' => "\033[31;31m%s\033[0m", + 'info' => "\033[0;34m%s\033[0m", ]; $format = '%s'; @@ -37,6 +38,8 @@ protected function debug(mixed $value, string $name = ''): void { } protected function printTitle(string $text, string $fill = '-', int $width = 80, string $cols_delim = '|', bool $has_content = FALSE): void { + $width = $this->getTuiWidth($width); + $this->printDivider($fill, $width, 'down'); $lines = explode(PHP_EOL, wordwrap($text, $width - 4, PHP_EOL)); foreach ($lines as $line) { @@ -47,6 +50,8 @@ protected function printTitle(string $text, string $fill = '-', int $width = 80, } protected function printSubtitle(string $text, string $fill = '=', int $width = 80): void { + $width = $this->getTuiWidth($width); + $is_multiline = strlen($text) + 4 >= $width; if ($is_multiline) { $this->printTitle($text, $fill, $width, 'both'); @@ -58,6 +63,8 @@ protected function printSubtitle(string $text, string $fill = '=', int $width = } protected function printDivider(string $fill = '-', int $width = 80, string $direction = 'none'): void { + $width = $this->getTuiWidth($width); + $start = $fill; $finish = $fill; switch ($direction) { @@ -81,6 +88,8 @@ protected function printDivider(string $fill = '-', int $width = 80, string $dir } protected function printBox(string $content, string $title = '', string $fill = 'ā”€', int $padding = 2, int $width = 80): void { + $width = $this->getTuiWidth($width); + $cols = 'ā”‚'; $max_width = $width - 2 - $padding * 2; @@ -132,7 +141,7 @@ protected function status(string $message, int $level = self::INSTALLER_STATUS_M break; case self::INSTALLER_STATUS_MESSAGE: - $prefix = 'iļø'; + $prefix = 'ā“˜ '; $color = 'info'; break; @@ -160,6 +169,8 @@ protected function status(string $message, int $level = self::INSTALLER_STATUS_M * Formatted values list. */ protected function formatValuesList(array $values, string $delim = '', int $width = 80): string { + $width = $this->getTuiWidth($width); + // Only keep the keys that are not numeric. $keys = array_filter(array_keys($values), static fn($key): bool => !is_numeric($key)); diff --git a/.vortex/installer/src/Traits/TuiTrait.php b/.vortex/installer/src/Traits/TuiTrait.php index 2e82c965a..edfae5420 100644 --- a/.vortex/installer/src/Traits/TuiTrait.php +++ b/.vortex/installer/src/Traits/TuiTrait.php @@ -27,6 +27,11 @@ trait TuiTrait { final const INSTALLER_STATUS_DEBUG = 3; + /** + * Width of the TUI. + */ + protected int $tuiWidth; + protected function ask(string $question, ?string $default, bool $close_handle = FALSE): ?string { if ($this->config->isQuiet()) { return $default; @@ -133,7 +138,7 @@ protected function printSummary(): void { $values['Vortex commit'] = $this->formatNotEmpty($this->config->get('VORTEX_INSTALL_COMMIT'), 'Latest'); $values[] = ''; - $values[] = str_repeat('ā”€', 80 - 2 - 2 * 2); + $values[] = str_repeat('ā”€', $this->getTuiWidth() - 2 - 2 * 2); $values[] = ''; $values['Name'] = $this->getAnswer('name'); @@ -166,7 +171,7 @@ protected function printSummary(): void { $values['Preserve docs in comments'] = $this->formatYesNo($this->getAnswer('preserve_doc_comments')); $values['Preserve Vortex comments'] = $this->formatYesNo($this->getAnswer('preserve_vortex_info')); - $content = $this->formatValuesList($values, '', 80 - 2 - 2 * 2); + $content = $this->formatValuesList($values, '', $this->getTuiWidth() - 2 - 2 * 2); $this->printBox($content, 'INSTALLATION SUMMARY'); } @@ -204,6 +209,15 @@ protected function commandExists(string $command): void { } } + protected function getTuiWidth(int $max = 80): int { + if (!isset($this->tuiWidth)) { + $width = intval($this->doExec('tput cols')); + $this->tuiWidth = $width > 0 ? $width : $max; + } + + return min($this->tuiWidth, $max); + } + /** * Execute command. * From c786bbeee9166988f34242469d17999ac4992870 Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 17:33:47 +1100 Subject: [PATCH 04/13] Fixed double footer in installer. --- .vortex/installer/src/Command/InstallCommand.php | 2 -- 1 file changed, 2 deletions(-) diff --git a/.vortex/installer/src/Command/InstallCommand.php b/.vortex/installer/src/Command/InstallCommand.php index edfffe7f7..dfb22119c 100644 --- a/.vortex/installer/src/Command/InstallCommand.php +++ b/.vortex/installer/src/Command/InstallCommand.php @@ -222,8 +222,6 @@ protected function doExecute(): void { $this->copyFiles(); $this->handleDemo(); - - $this->printFooter(); } protected function collectAnswers(): void { From e62382e6be68b8ec5369bae45532a409c9ad459b Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 17:34:01 +1100 Subject: [PATCH 05/13] Fixed dst dir not created by the installer. --- .vortex/installer/src/Command/InstallCommand.php | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/.vortex/installer/src/Command/InstallCommand.php b/.vortex/installer/src/Command/InstallCommand.php index dfb22119c..87b15dee5 100644 --- a/.vortex/installer/src/Command/InstallCommand.php +++ b/.vortex/installer/src/Command/InstallCommand.php @@ -161,8 +161,17 @@ protected function resolveOptions(array $options, ?string $path): void { // Set destination directory. if (!empty($path)) { $path = $this->fsGetAbsolutePath($path); - if (!is_readable($path) || !is_dir($path)) { - throw new \RuntimeException(sprintf('Destination directory "%s" is not readable or does not exist.', $path)); + + if (file_exists($path)) { + if (is_file($path)) { + throw new \RuntimeException(sprintf('Destination directory "%s" is a file.', $path)); + } + } + else { + $this->fs->mkdir($path); + if (!is_readable($path) || !is_dir($path)) { + throw new \RuntimeException(sprintf('Destination directory "%s" is not readable or does not exist.', $path)); + } } } $this->config->set('VORTEX_INSTALL_DST_DIR', $path ?: static::getenvOrDefault('VORTEX_INSTALL_DST_DIR', $this->fsGetRootDir())); From 7874336f59968c2edf2dc08d931fe143f079f88e Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 17:34:23 +1100 Subject: [PATCH 06/13] Fixed release discovery in installer. --- .vortex/installer/src/Traits/DownloadTrait.php | 12 +++++++++--- 1 file changed, 9 insertions(+), 3 deletions(-) diff --git a/.vortex/installer/src/Traits/DownloadTrait.php b/.vortex/installer/src/Traits/DownloadTrait.php index 604c816f6..48831292d 100644 --- a/.vortex/installer/src/Traits/DownloadTrait.php +++ b/.vortex/installer/src/Traits/DownloadTrait.php @@ -57,7 +57,7 @@ protected function downloadScaffoldRemote(): void { $this->status('Done', self::INSTALLER_STATUS_SUCCESS); } - protected function findLatestVortexRelease(string $org, string $project, string $release_prefix): ?string { + protected function findLatestVortexRelease(string $org, string $project, ?string $release_prefix): ?string { $release_url = sprintf('https://api.github.com/repos/%s/%s/releases', $org, $project); $release_contents = file_get_contents($release_url, FALSE, stream_context_create([ 'http' => ['method' => 'GET', 'header' => ['User-Agent: PHP']], @@ -68,9 +68,15 @@ protected function findLatestVortexRelease(string $org, string $project, string } $records = json_decode($release_contents, TRUE); + + if (!$release_prefix) { + return is_scalar($records[0]['tag_name']) ? strval($records[0]['tag_name']) : NULL; + } + foreach ($records as $record) { - if (isset($record['tag_name']) && ($release_prefix && str_contains((string) $record['tag_name'], $release_prefix) || !$release_prefix)) { - return $record['tag_name']; + $tag_name = is_scalar($record['tag_name']) ? strval($record['tag_name']) : ''; + if (str_contains($tag_name, $release_prefix)) { + return $tag_name; } } From bc06d9da3e373d196dc729cb1264333c75bb59bc Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 18:59:52 +1100 Subject: [PATCH 07/13] Added logo to installer. --- .vortex/installer/src/Traits/PrinterTrait.php | 50 +++++++++++++------ .vortex/installer/src/Traits/TuiTrait.php | 20 ++++++++ 2 files changed, 54 insertions(+), 16 deletions(-) diff --git a/.vortex/installer/src/Traits/PrinterTrait.php b/.vortex/installer/src/Traits/PrinterTrait.php index acda65dd8..773a9ebea 100644 --- a/.vortex/installer/src/Traits/PrinterTrait.php +++ b/.vortex/installer/src/Traits/PrinterTrait.php @@ -10,23 +10,32 @@ trait PrinterTrait { protected function out(string $text, ?string $color = NULL, bool $new_line = TRUE): void { - $styles = [ + $text = $this->config->get('ANSI') ? $this->formatColor($text, $color) : $text; + + if ($new_line) { + $text .= PHP_EOL; + } + + print $text; + } + + protected function formatColor(string $text, string $color): string { + $colors = [ + 'green' => "\033[0;32m%s\033[0m", + 'red' => "\033[0;31m%s\033[0m", + 'blue' => "\033[0;34m%s\033[0m", + 'yellow' => "\033[0;33m%s\033[0m", + 'cyan' => "\033[0;36m%s\033[0m", + 'magenta' => "\033[0;35m%s\033[0m", + 'white' => "\033[0;37m%s\033[0m", 'success' => "\033[0;32m%s\033[0m", 'error' => "\033[31;31m%s\033[0m", 'info' => "\033[0;34m%s\033[0m", ]; - $format = '%s'; - - if (isset($styles[$color]) && $this->config->get('ANSI')) { - $format = $styles[$color]; - } + $format = $colors[$color] ?: '%s'; - if ($new_line) { - $format .= PHP_EOL; - } - - printf($format, $text); + return sprintf($format, $text); } protected function debug(mixed $value, string $name = ''): void { @@ -43,8 +52,8 @@ protected function printTitle(string $text, string $fill = '-', int $width = 80, $this->printDivider($fill, $width, 'down'); $lines = explode(PHP_EOL, wordwrap($text, $width - 4, PHP_EOL)); foreach ($lines as $line) { - $line = ' ' . $line . ' '; - print $cols_delim . str_pad($line, $width - 2, ' ', STR_PAD_BOTH) . $cols_delim . PHP_EOL; + $line = ' ' . $this->formatBold($line) . ' '; + print $cols_delim . str_pad($line, $width - 2 + (strlen($line) - $this->getVisibleLength($line)), ' ', STR_PAD_BOTH) . $cols_delim . PHP_EOL; } $this->printDivider($fill, $width, $has_content ? 'up' : 'both'); } @@ -90,12 +99,10 @@ protected function printDivider(string $fill = '-', int $width = 80, string $dir protected function printBox(string $content, string $title = '', string $fill = 'ā”€', int $padding = 2, int $width = 80): void { $width = $this->getTuiWidth($width); - $cols = 'ā”‚'; - $max_width = $width - 2 - $padding * 2; $lines = explode(PHP_EOL, wordwrap(rtrim($content, PHP_EOL), $max_width, PHP_EOL)); $pad = str_pad(' ', $padding); - $mask = sprintf('%s%s%%-%ss%s%s', $cols, $pad, $max_width, $pad, $cols) . PHP_EOL; + $mask = sprintf('ā”‚%s%%-%ss%sā”‚', $pad, $max_width, $pad) . PHP_EOL; print PHP_EOL; if (!empty($title)) { @@ -229,4 +236,15 @@ protected function formatNotEmpty(mixed $value, mixed $default): mixed { return empty($value) ? $default : $value; } + protected function formatBold(string $text): string { + return "\033[1m" . $text . "\033[0m"; + } + + protected function getVisibleLength(string $text): int { + // Remove ANSI escape sequences using a regex. + $plain_text = preg_replace('/\033\[[0-9;]*m/', '', $text); + + return mb_strlen($plain_text); + } + } diff --git a/.vortex/installer/src/Traits/TuiTrait.php b/.vortex/installer/src/Traits/TuiTrait.php index edfae5420..b6e773739 100644 --- a/.vortex/installer/src/Traits/TuiTrait.php +++ b/.vortex/installer/src/Traits/TuiTrait.php @@ -73,6 +73,26 @@ protected function closeStdinHandle(): void { } protected function printHeader(): void { + $logo = <<getTuiWidth() >= 80) { + $this->out($logo, 'green'); + } + if ($this->config->isQuiet()) { $this->printHeaderQuiet(); } From f2f10050b2c4446358fcfb5a94ce4456ff92e561 Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 19:17:13 +1100 Subject: [PATCH 08/13] Fixed ANSI argument not correctly used in installer. --- .vortex/installer/src/Command/InstallCommand.php | 4 ++-- .vortex/installer/src/Traits/PrinterTrait.php | 8 ++++++-- 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/.vortex/installer/src/Command/InstallCommand.php b/.vortex/installer/src/Command/InstallCommand.php index 87b15dee5..7ca0f1a92 100644 --- a/.vortex/installer/src/Command/InstallCommand.php +++ b/.vortex/installer/src/Command/InstallCommand.php @@ -143,8 +143,8 @@ protected function resolveOptions(array $options, ?string $path): void { $this->config->set('quiet', TRUE); } - if (!empty($options['no-ansi'])) { - $this->config->set('ANSI', FALSE); + if (isset($options['ansi']) && !is_null($options['ansi'])) { + $this->config->set('ANSI', $options['ansi']); } else { // On Windows, default to no ANSI, except in ANSICON and ConEmu. diff --git a/.vortex/installer/src/Traits/PrinterTrait.php b/.vortex/installer/src/Traits/PrinterTrait.php index 773a9ebea..dad8b059c 100644 --- a/.vortex/installer/src/Traits/PrinterTrait.php +++ b/.vortex/installer/src/Traits/PrinterTrait.php @@ -10,7 +10,7 @@ trait PrinterTrait { protected function out(string $text, ?string $color = NULL, bool $new_line = TRUE): void { - $text = $this->config->get('ANSI') ? $this->formatColor($text, $color) : $text; + $text = $color ? $this->formatColor($text, $color) : $text; if ($new_line) { $text .= PHP_EOL; @@ -20,6 +20,10 @@ protected function out(string $text, ?string $color = NULL, bool $new_line = TRU } protected function formatColor(string $text, string $color): string { + if (!$this->config->get('ANSI')) { + return $text; + } + $colors = [ 'green' => "\033[0;32m%s\033[0m", 'red' => "\033[0;31m%s\033[0m", @@ -33,7 +37,7 @@ protected function formatColor(string $text, string $color): string { 'info' => "\033[0;34m%s\033[0m", ]; - $format = $colors[$color] ?: '%s'; + $format = $colors[$color] ?? '%s'; return sprintf($format, $text); } From e734116c7236797b64c1a1ed53125490fbae09b5 Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 19:18:08 +1100 Subject: [PATCH 09/13] Added colours to questions of the installer. --- .vortex/installer/src/Traits/TuiTrait.php | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/.vortex/installer/src/Traits/TuiTrait.php b/.vortex/installer/src/Traits/TuiTrait.php index b6e773739..3b789239b 100644 --- a/.vortex/installer/src/Traits/TuiTrait.php +++ b/.vortex/installer/src/Traits/TuiTrait.php @@ -37,20 +37,23 @@ protected function ask(string $question, ?string $default, bool $close_handle = return $default; } - $question = sprintf('> %s [%s] ', $question, $default); + $this->out(sprintf('%s [%s] ', $this->formatColor('> ' .$question, 'green'), $this->formatColor($default, 'yellow')), NULL, FALSE); - $this->out($question, 'question', FALSE); $handle = $this->getStdinHandle(); $answer = fgets($handle); if ($answer !== FALSE) { $answer = trim($answer); } + $answer = empty($answer) ? $default : $answer; + + $this->out($answer, 'cyan'); + if ($close_handle) { $this->closeStdinHandle(); } - return empty($answer) ? $default : $answer; + return $answer; } protected function getStdinHandle(): mixed { @@ -300,7 +303,7 @@ protected function askShouldProceed(): bool { $proceed = self::ANSWER_YES; if (!$this->config->isQuiet()) { - $proceed = $this->ask(sprintf('Proceed with installing Vortex into your project\'s directory "%s"? (Y,n)', $this->config->getDstDir()), $proceed, TRUE); + $proceed = $this->ask(sprintf('Proceed with installing Vortex into your project\'s directory "%s"?', $this->config->getDstDir()), $proceed, TRUE); } // Kill-switch to not proceed with install. If false, the install will not From 1bdbbecdc06002d82e4a65d52049ca91d69d9318 Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 19:30:02 +1100 Subject: [PATCH 10/13] Fixed installation result footer in the installer for a new install. Fixed coding standards. --- .vortex/installer/src/Command/InstallCommand.php | 2 +- .vortex/installer/src/Traits/TuiTrait.php | 15 +++++++++++---- 2 files changed, 12 insertions(+), 5 deletions(-) diff --git a/.vortex/installer/src/Command/InstallCommand.php b/.vortex/installer/src/Command/InstallCommand.php index 7ca0f1a92..544fbfe94 100644 --- a/.vortex/installer/src/Command/InstallCommand.php +++ b/.vortex/installer/src/Command/InstallCommand.php @@ -143,7 +143,7 @@ protected function resolveOptions(array $options, ?string $path): void { $this->config->set('quiet', TRUE); } - if (isset($options['ansi']) && !is_null($options['ansi'])) { + if (!is_null($options['ansi'])) { $this->config->set('ANSI', $options['ansi']); } else { diff --git a/.vortex/installer/src/Traits/TuiTrait.php b/.vortex/installer/src/Traits/TuiTrait.php index 3b789239b..4724e5e98 100644 --- a/.vortex/installer/src/Traits/TuiTrait.php +++ b/.vortex/installer/src/Traits/TuiTrait.php @@ -32,12 +32,17 @@ trait TuiTrait { */ protected int $tuiWidth; + /** + * Is Vortex initially installed. + */ + protected bool $isInitiallyInstalled = FALSE; + protected function ask(string $question, ?string $default, bool $close_handle = FALSE): ?string { if ($this->config->isQuiet()) { return $default; } - $this->out(sprintf('%s [%s] ', $this->formatColor('> ' .$question, 'green'), $this->formatColor($default, 'yellow')), NULL, FALSE); + $this->out(sprintf('%s [%s] ', $this->formatColor('> ' . $question, 'green'), $this->formatColor($default, 'yellow')), NULL, FALSE); $handle = $this->getStdinHandle(); $answer = fgets($handle); @@ -96,6 +101,8 @@ protected function printHeader(): void { $this->out($logo, 'green'); } + $this->isInitiallyInstalled = $this->isInstalled(); + if ($this->config->isQuiet()) { $this->printHeaderQuiet(); } @@ -116,7 +123,7 @@ protected function printHeaderInteractive(): void { $content .= sprintf('This will install Vortex into your project at commit "%s".', $commit) . PHP_EOL; } $content .= PHP_EOL; - if ($this->isInstalled()) { + if ($this->isInitiallyInstalled) { $content .= 'It looks like Vortex is already installed into this project.' . PHP_EOL; $content .= PHP_EOL; } @@ -142,7 +149,7 @@ protected function printHeaderQuiet(): void { } $content .= PHP_EOL; - if ($this->isInstalled()) { + if ($this->isInitiallyInstalled) { $content .= 'It looks like Vortex is already installed into this project.' . PHP_EOL; $content .= PHP_EOL; } @@ -206,7 +213,7 @@ protected function printAbort(): void { protected function printFooter(): void { print PHP_EOL; - if ($this->isInstalled()) { + if ($this->isInitiallyInstalled) { $this->printBox('Finished updating Vortex. Review changes and commit required files.'); } else { From df49e801b05aad53c37619e9c511f535adb1cf37 Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 19:57:45 +1100 Subject: [PATCH 11/13] Updated variables description in `.env` file. --- .env | 92 ++++++++++--------- .vortex/docs/content/workflows/variables.mdx | 31 +++++-- .vortex/installer/composer.json | 2 +- .vortex/installer/composer.lock | 33 ++++--- .../installer/src/Command/InstallCommand.php | 7 ++ .vortex/installer/src/Traits/PromptsTrait.php | 2 +- 6 files changed, 100 insertions(+), 67 deletions(-) diff --git a/.env b/.env index df0538b61..85315c969 100644 --- a/.env +++ b/.env @@ -27,7 +27,7 @@ # the project directory and can be overridden with $COMPOSE_PROJECT_NAME. VORTEX_PROJECT=your_site -# Name of the webroot directory with Drupal codebase. +# Name of the web root directory containing a Drupal codebase. VORTEX_WEBROOT=web # The timezone for the containers. @@ -43,11 +43,13 @@ DRUPAL_PROFILE=standard # Drupal theme name. DRUPAL_THEME=your_site_theme -# Stage file proxy origin. Note that HTTP Auth provided by Shield will be -# automatically added to the origin URL. +# Stage file proxy origin. +# +# If using Shield, the HTTP authentication credentials will be automatically +# added to the origin URL. DRUPAL_STAGE_FILE_PROXY_ORIGIN=https://www.your-site-url.example/ -# Shield print message. +# Shield message. DRUPAL_SHIELD_PRINT="Restricted access." #;< REDIS @@ -71,13 +73,16 @@ DRUPAL_CLAMAV_MODE=daemon # PROVISION # ################################################################################ -# Variables in this section control how the site is provisioned. +# By "provision", we mean the process of initialising the database (from dump +# or fresh install from profile), running updates, appying configuration +# changes, clearing caches and performing other tasks that prepare the site for +# use. # @see https://vortex.drevops.com/workflows/provision # Set to 1 to install a site from profile instead of the database file dump. VORTEX_PROVISION_USE_PROFILE=0 -# Overwrite existing database if it exists. +# Overwrite a database if it exists. # # Usually set to 0 in deployed environments and can be temporary set to 1 for # a specific deployment. @@ -90,23 +95,42 @@ VORTEX_PROVISION_OVERRIDE_DB=0 # environments and is always skipped in the production environment. VORTEX_PROVISION_SANITIZE_DB_SKIP=0 -# Sanitization email pattern. Sanitization is enabled by default in all -# non-production environments. +# Sanitization email pattern. +# +# Applied if database sanitization is enabled. # @see https://vortex.drevops.com/workflows/build#sanitization VORTEX_PROVISION_SANITIZE_DB_EMAIL="user_%uid@your-site-url.example" # Put the site into a maintenance mode during site provisioning. VORTEX_PROVISION_USE_MAINTENANCE_MODE=1 +################################################################################ +# HOSTING # +################################################################################ + +#;< LAGOON +# Lagoon project name. May be different from $VORTEX_PROJECT. +LAGOON_PROJECT=your_site + +# Dedicated branch to identify the production environment. +VORTEX_LAGOON_PRODUCTION_BRANCH=main +#;> LAGOON + +#;< ACQUIA +# Acquia application name. +VORTEX_ACQUIA_APP_NAME= +#;> ACQUIA + ################################################################################ # DATABASE # ################################################################################ # Database service runs a single database within a container. -# See settings.php for database credentials defaults. +# See settings.php for database credentials defaults or run +# `ahoy drush sql:connect`. -# Database can be imported from a file dump into an empty database started from -# the database default image or can exist in a pre-built container image. +# Database can be imported from a *file dump* into an empty database started +# from the database default image or can *exist* in a pre-built container image. # Defaults to importing from a file. # @see https://vortex.drevops.com/workflows/database @@ -124,26 +148,28 @@ VORTEX_DB_FILE=db.sql #;< !PROVISION_USE_PROFILE # Database can be sourced from one of the following locations: # -# - `curl` - directly from URL as a file using CURL. -# - `ftp` - directly from FTP as a file using CURL. +# - `curl` - directly from a URL as a file using CURL. +# - `ftp` - directly from an FTP as a file using CURL. # - `acquia` - from the latest Acquia backup via Cloud API as a file. # - `lagoon` - from Lagoon main environment as a file. # - `container_registry` - from the container registry as a container image. # - `none` - not downloaded, site is freshly installed for every build. - +# # Note that "container_registry" works only for database-in-image # database storage (when $VORTEX_DB_IMAGE variable has a value). VORTEX_DB_DOWNLOAD_SOURCE=curl #;< VORTEX_DB_DOWNLOAD_SOURCE_CURL -# Database dump file sourced from CURL, with optional HTTP Basic Authentication -# credentials embedded into the value. +# Database dump file sourced from CURL. +# +# HTTP Basic Authentication credentials should be embedded into the value. VORTEX_DB_DOWNLOAD_CURL_URL= #;> VORTEX_DB_DOWNLOAD_SOURCE_CURL # Environment to download the database from. +# +# Applies to Acquia and Lagoon environments. VORTEX_DB_DOWNLOAD_ENVIRONMENT=prod -#;> !PROVISION_USE_PROFILE #;< FTP # Database dump file sourced from FTP. @@ -159,17 +185,11 @@ VORTEX_DB_DOWNLOAD_FTP_FILE=db.sql #;> FTP #;< ACQUIA -# Acquia application name to download the database from. -VORTEX_ACQUIA_APP_NAME= - # Acquia database name to download the database from. VORTEX_DB_DOWNLOAD_ACQUIA_DB_NAME=your_site #;> ACQUIA -#;< LAGOON -# Lagoon project name. May be different from $VORTEX_PROJECT. -LAGOON_PROJECT=your_site -#;> LAGOON +#;> !PROVISION_USE_PROFILE #;< DEPLOYMENT ################################################################################ @@ -186,12 +206,6 @@ LAGOON_PROJECT=your_site # # See https://vortex.drevops.com/workflows/deploy VORTEX_DEPLOY_TYPES=artifact - -#;< LAGOON -# Dedicated branch to identify the production environment. -VORTEX_LAGOON_PRODUCTION_BRANCH=main -#;> LAGOON - #;> DEPLOYMENT #;< NOTIFICATIONS @@ -222,16 +236,12 @@ VORTEX_NOTIFY_EMAIL_RECIPIENTS="webmaster@your-site-url.example" ################################################################################ # DEMO # ################################################################################ - -#------------------------------------------------------------------------------- -# Variables used to override project-specific values for demonstration purposes. -# -# This is to allow to demonstrate how Vortex works without asking users -# to perform any additional steps when trying out Vortex for the first time. -# -# Remove this section from your project when database download integration setup -# is finished. -#------------------------------------------------------------------------------- +# # +# Override project-specific values for demonstration purposes. # +# Used to showcase Vortex without asking users to perform additional steps. # +# Remove this section after completing database download integration. # +# # +################################################################################ #;< VORTEX_DB_DOWNLOAD_SOURCE_CURL # URL of the demo database used for demonstration with CURL database @@ -240,7 +250,7 @@ VORTEX_DB_DOWNLOAD_CURL_URL=https://github.com/drevops/vortex/releases/download/ #;> VORTEX_DB_DOWNLOAD_SOURCE_CURL #;< VORTEX_DB_IMAGE -# The name of the demo database image used for demonstration only for +# The name of the demo database image used for demonstration for # database-in-image data storage. #; The line below will be automatically uncommented for database-in-image #; storage. It is commented out to allow running non-database-in-image diff --git a/.vortex/docs/content/workflows/variables.mdx b/.vortex/docs/content/workflows/variables.mdx index dfffd05a4..9442d6b59 100644 --- a/.vortex/docs/content/workflows/variables.mdx +++ b/.vortex/docs/content/workflows/variables.mdx @@ -162,7 +162,7 @@ Defined in: `.env` ### `DRUPAL_SHIELD_PRINT` -Shield print message. +Shield message. Default value: `Restricted access.` @@ -186,7 +186,9 @@ Defined in: `.env`, `scripts/vortex/provision.sh` ### `DRUPAL_STAGE_FILE_PROXY_ORIGIN` -Stage file proxy origin. Note that HTTP Auth provided by Shield will be
automatically added to the origin URL. +Stage file proxy origin. + +If using Shield, the HTTP authentication credentials will be automatically
added to the origin URL. Default value: `https://www.your-site-url.example/` @@ -294,7 +296,7 @@ Defined in: `scripts/vortex/task-purge-cache-acquia.sh` ### `VORTEX_ACQUIA_APP_NAME` -Acquia application name to download the database from. +Acquia application name. Default value: `UNDEFINED` @@ -488,7 +490,9 @@ Defined in: `.env`, `scripts/vortex/download-db-acquia.sh` ### `VORTEX_DB_DOWNLOAD_CURL_URL` -Database dump file sourced from CURL, with optional HTTP Basic Authentication
credentials embedded into the value. +Database dump file sourced from CURL. + +HTTP Basic Authentication credentials should be embedded into the value. Default value: `UNDEFINED` @@ -498,6 +502,8 @@ Defined in: `.env`, `scripts/vortex/download-db-curl.sh` Environment to download the database from. +Applies to Acquia and Lagoon environments. + Default value: `prod` Defined in: `.env`, `scripts/vortex/download-db-acquia.sh`, `scripts/vortex/download-db-lagoon.sh` @@ -618,6 +624,15 @@ Defined in: `scripts/vortex/download-db-lagoon.sh` ### `VORTEX_DB_DOWNLOAD_SOURCE` +Database can be sourced from one of the following locations: + +- `curl` - directly from a URL as a file using CURL. +- `ftp` - directly from an FTP as a file using CURL. +- `acquia` - from the latest Acquia backup via Cloud API as a file. +- `lagoon` - from Lagoon main environment as a file. +- `container_registry` - from the container registry as a container image. +- `none` - not downloaded, site is freshly installed for every build. + Note that "container_registry" works only for database-in-image
database storage (when [`$VORTEX_DB_IMAGE`](#vortex_db_image) variable has a value). Default value: `curl` @@ -1533,7 +1548,7 @@ Defined in: `scripts/vortex/provision.sh` ### `VORTEX_PROVISION_OVERRIDE_DB` -Overwrite existing database if it exists. +Overwrite a database if it exists. Usually set to `0` in deployed environments and can be temporary set to `1` for
a specific deployment.
Set this to `1` in .env.local to override when developing locally. @@ -1561,7 +1576,9 @@ Defined in: `scripts/vortex/provision-sanitize-db.sh` ### `VORTEX_PROVISION_SANITIZE_DB_EMAIL` -Sanitization email pattern. Sanitization is enabled by default in all
non-production environments.
@see https://vortex.drevops.com/workflows/build#sanitization +Sanitization email pattern. + +Applied if database sanitization is enabled.
@see https://vortex.drevops.com/workflows/build#sanitization Default value: `user_%uid@your-site-url.example` @@ -1862,7 +1879,7 @@ Defined in: `.env` ### `VORTEX_WEBROOT` -Name of the webroot directory with Drupal codebase. +Name of the web root directory containing a Drupal codebase. Default value: `web` diff --git a/.vortex/installer/composer.json b/.vortex/installer/composer.json index 472178270..e5d246416 100644 --- a/.vortex/installer/composer.json +++ b/.vortex/installer/composer.json @@ -27,7 +27,7 @@ "drupal/coder": "^8.3", "ergebnis/composer-normalize": "^2.45", "mikey179/vfsstream": "^1.6", - "opis/closure": "^3.6", + "opis/closure": "^4", "phpstan/phpstan": "^1.10", "phpunit/phpunit": "^10", "rector/rector": "^1.0.0" diff --git a/.vortex/installer/composer.lock b/.vortex/installer/composer.lock index 79fe7d405..25c0393e9 100644 --- a/.vortex/installer/composer.lock +++ b/.vortex/installer/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "5a995c729a9bc0de421a808833578ca6", + "content-hash": "9b1ac86bcb8602ff6b784952c86facaf", "packages": [ { "name": "psr/container", @@ -627,12 +627,12 @@ }, "type": "library", "extra": { + "thanks": { + "url": "https://github.com/symfony/contracts", + "name": "symfony/contracts" + }, "branch-alias": { "dev-main": "3.5-dev" - }, - "thanks": { - "name": "symfony/contracts", - "url": "https://github.com/symfony/contracts" } }, "autoload": { @@ -1896,34 +1896,33 @@ }, { "name": "opis/closure", - "version": "3.6.3", + "version": "4.3.1", "source": { "type": "git", "url": "https://github.com/opis/closure.git", - "reference": "3d81e4309d2a927abbe66df935f4bb60082805ad" + "reference": "9b6ef5f622b4b29cf98bdc144f2a370c3a40d4df" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/opis/closure/zipball/3d81e4309d2a927abbe66df935f4bb60082805ad", - "reference": "3d81e4309d2a927abbe66df935f4bb60082805ad", + "url": "https://api.github.com/repos/opis/closure/zipball/9b6ef5f622b4b29cf98bdc144f2a370c3a40d4df", + "reference": "9b6ef5f622b4b29cf98bdc144f2a370c3a40d4df", "shasum": "" }, "require": { - "php": "^5.4 || ^7.0 || ^8.0" + "php": "^8.0" }, "require-dev": { - "jeremeamia/superclosure": "^2.0", - "phpunit/phpunit": "^4.0 || ^5.0 || ^6.0 || ^7.0 || ^8.0 || ^9.0" + "phpunit/phpunit": "^9.0" }, "type": "library", "extra": { "branch-alias": { - "dev-master": "3.6.x-dev" + "dev-master": "4.x-dev" } }, "autoload": { "files": [ - "functions.php" + "src/functions.php" ], "psr-4": { "Opis\\Closure\\": "src/" @@ -1943,7 +1942,7 @@ "email": "sarca_sorin@hotmail.com" } ], - "description": "A library that can be used to serialize closures (anonymous functions) and arbitrary objects.", + "description": "A library that can be used to serialize closures (anonymous functions) and arbitrary data.", "homepage": "https://opis.io/closure", "keywords": [ "anonymous functions", @@ -1955,9 +1954,9 @@ ], "support": { "issues": "https://github.com/opis/closure/issues", - "source": "https://github.com/opis/closure/tree/3.6.3" + "source": "https://github.com/opis/closure/tree/4.3.1" }, - "time": "2022-01-27T09:35:39+00:00" + "time": "2025-01-10T20:42:24+00:00" }, { "name": "phar-io/manifest", diff --git a/.vortex/installer/src/Command/InstallCommand.php b/.vortex/installer/src/Command/InstallCommand.php index 544fbfe94..b4310a56d 100644 --- a/.vortex/installer/src/Command/InstallCommand.php +++ b/.vortex/installer/src/Command/InstallCommand.php @@ -96,6 +96,13 @@ protected function configure(): void { protected function execute(InputInterface $input, OutputInterface $output): int { $this->output = $output; + // @see https://github.com/drevops/vortex/issues/1502 + if ($input->getOption('help') || $input->getArgument('path') == 'help') { + $output->write($this->getHelp()); + + return Command::SUCCESS; + } + try { $this->checkRequirements(); diff --git a/.vortex/installer/src/Traits/PromptsTrait.php b/.vortex/installer/src/Traits/PromptsTrait.php index ccf8930c0..6bd181a60 100644 --- a/.vortex/installer/src/Traits/PromptsTrait.php +++ b/.vortex/installer/src/Traits/PromptsTrait.php @@ -307,7 +307,7 @@ protected function processDemoMode(string $dir): void { if ($this->getAnswer('provision_use_profile') === self::ANSWER_NO) { $download_source = $this->getAnswer('database_download_source'); $db_file = static::getenvOrDefault('VORTEX_DB_DIR', './.data') . DIRECTORY_SEPARATOR . static::getenvOrDefault('VORTEX_DB_FILE', 'db.sql'); - $has_comment = File::fileContains('to allow to demonstrate how Vortex works without', $this->config->getDstDir() . '/.env'); + $has_comment = File::fileContains('Override project-specific values for demonstration purposes', $this->config->getDstDir() . '/.env'); // Enable Vortex demo mode if download source is file AND // there is no downloaded file present OR if there is a demo comment in From 9c31148284f3f68060433012ca8c8638aae656a0 Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 22:19:23 +1100 Subject: [PATCH 12/13] [#1476] Added removal of the multiple empty lines in installer. --- .vortex/installer/src/Command/InstallCommand.php | 1 + .vortex/installer/src/Traits/PromptsTrait.php | 8 +++++++- 2 files changed, 8 insertions(+), 1 deletion(-) diff --git a/.vortex/installer/src/Command/InstallCommand.php b/.vortex/installer/src/Command/InstallCommand.php index b4310a56d..928e8cd8e 100644 --- a/.vortex/installer/src/Command/InstallCommand.php +++ b/.vortex/installer/src/Command/InstallCommand.php @@ -381,6 +381,7 @@ protected function replaceTokens(): void { 'preserve_vortex_info', 'vortex_internal', 'enable_commented_code', + 'empty_lines', ]; foreach ($processors as $name) { diff --git a/.vortex/installer/src/Traits/PromptsTrait.php b/.vortex/installer/src/Traits/PromptsTrait.php index 6bd181a60..727eb419a 100644 --- a/.vortex/installer/src/Traits/PromptsTrait.php +++ b/.vortex/installer/src/Traits/PromptsTrait.php @@ -375,10 +375,16 @@ protected function processVortexInternal(string $dir): void { } protected function processEnableCommentedCode(string $dir): void { - // Enable_commented_code. File::dirReplaceContent('##### ', '', $dir); } + protected function processEmptyLines(string $dir): void { + $files = File::scandirRecursive($dir, File::ignorePaths()); + foreach ($files as $filename) { + File::fileReplaceContent('/(\n\s*\n)+/', "\n\n", $filename); + } + } + protected function processWebroot(string $dir): void { $new_name = $this->getAnswer('webroot', 'web'); From 019a204d0dfc9f0d13f61f18119d1f218e0b57b1 Mon Sep 17 00:00:00 2001 From: Alex Skrypnyk Date: Sun, 19 Jan 2025 22:01:51 +1100 Subject: [PATCH 13/13] [#1746] Fixed domain URL replacements in the installer. --- .circleci/config.yml | 2 +- .env | 8 +- .github/workflows/renovate.yml | 2 +- .lagoon.yml | 8 +- .../.utils/variables/extra/.env.variables.sh | 4 +- .../.utils/variables/extra/ci.variables.sh | 2 +- .vortex/docs/content/workflows/variables.mdx | 14 +-- .../installer/src/Command/InstallCommand.php | 2 +- .vortex/installer/src/Traits/PromptsTrait.php | 101 +++++++----------- .vortex/installer/src/Traits/TuiTrait.php | 2 +- .vortex/tests/bats/_helper.bash | 2 +- .vortex/tests/bats/install.parameters.bats | 21 ++-- hooks/library/domains.txt | 2 +- 13 files changed, 70 insertions(+), 100 deletions(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index e40092ecd..96631c0c9 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -39,7 +39,7 @@ aliases: #;< SCAFFOLD_DEV - &renovatebot_git_author 'Renovate Self Hosted ' #;> SCAFFOLD_DEV - ##### - &renovatebot_git_author 'RenovateBot Self Hosted ' + ##### - &renovatebot_git_author 'RenovateBot Self Hosted ' # The schedule to run RenovateBot on. Defaults to running twice a day. - &renovatebot_schedule "5 11,23 * * *" #;> RENOVATEBOT diff --git a/.env b/.env index 85315c969..9cacbb462 100644 --- a/.env +++ b/.env @@ -47,7 +47,7 @@ DRUPAL_THEME=your_site_theme # # If using Shield, the HTTP authentication credentials will be automatically # added to the origin URL. -DRUPAL_STAGE_FILE_PROXY_ORIGIN=https://www.your-site-url.example/ +DRUPAL_STAGE_FILE_PROXY_ORIGIN=https://www.your-site-domain.example/ # Shield message. DRUPAL_SHIELD_PRINT="Restricted access." @@ -99,7 +99,7 @@ VORTEX_PROVISION_SANITIZE_DB_SKIP=0 # # Applied if database sanitization is enabled. # @see https://vortex.drevops.com/workflows/build#sanitization -VORTEX_PROVISION_SANITIZE_DB_EMAIL="user_%uid@your-site-url.example" +VORTEX_PROVISION_SANITIZE_DB_EMAIL="user_%uid@your-site-domain.example" # Put the site into a maintenance mode during site provisioning. VORTEX_PROVISION_USE_MAINTENANCE_MODE=1 @@ -222,14 +222,14 @@ VORTEX_DEPLOY_TYPES=artifact VORTEX_NOTIFY_CHANNELS=email # Email to send notifications from. -VORTEX_NOTIFY_EMAIL_FROM="webmaster@your-site-url.example" +VORTEX_NOTIFY_EMAIL_FROM="webmaster@your-site-domain.example" # Email address(es) to send notifications to. # # Multiple names can be specified as a comma-separated list of email addresses # with optional names in the format "email|name". # Example: "to1@example.com|Jane Doe, to2@example.com|John Doe" -VORTEX_NOTIFY_EMAIL_RECIPIENTS="webmaster@your-site-url.example" +VORTEX_NOTIFY_EMAIL_RECIPIENTS="webmaster@your-site-domain.example" #;> NOTIFICATIONS #;< DEMO diff --git a/.github/workflows/renovate.yml b/.github/workflows/renovate.yml index 43376e915..8d326bd11 100644 --- a/.github/workflows/renovate.yml +++ b/.github/workflows/renovate.yml @@ -25,4 +25,4 @@ jobs: RENOVATE_DEPENDENCY_DASHBOARD_TITLE: 'Renovate Dependency Dashboard (self-hosted) by GitHub Actions' RENOVATE_DEPENDENCY_DASHBOARD: ${{ vars.RENOVATE_DEPENDENCY_DASHBOARD || 'false' }} RENOVATE_DRY_RUN: ${{ vars.RENOVATE_DRY_RUN || 'false' }} - RENOVATE_GIT_AUTHOR: ${{ vars.RENOVATE_GIT_AUTHOR || 'Renovate Self Hosted ' }} + RENOVATE_GIT_AUTHOR: ${{ vars.RENOVATE_GIT_AUTHOR || 'Renovate Self Hosted ' }} diff --git a/.lagoon.yml b/.lagoon.yml index 2302c9af9..cd5de244e 100644 --- a/.lagoon.yml +++ b/.lagoon.yml @@ -109,10 +109,10 @@ environments: # By default, Lagoon installs Let's Encrypt certificates for all environments. # routes: # - nginx-php: - # - www.your-site-url.example: + # - www.your-site-domain.example: # tls-acme: 'false' monitoring_urls: - - www.your-site-url.example + - www.your-site-domain.example # Branch name that represents development environment. develop: cronjobs: @@ -124,10 +124,10 @@ environments: # By default, Lagoon installs Let's Encrypt certificates for all environments. # routes: # - nginx-php: - # - dev.your-site-url.example: + # - dev.your-site-domain.example: # tls-acme: 'false' monitoring_urls: - - dev.your-site-url.example + - dev.your-site-domain.example routes: insecure: Redirect diff --git a/.vortex/docs/.utils/variables/extra/.env.variables.sh b/.vortex/docs/.utils/variables/extra/.env.variables.sh index d291ca9ea..982c12512 100755 --- a/.vortex/docs/.utils/variables/extra/.env.variables.sh +++ b/.vortex/docs/.utils/variables/extra/.env.variables.sh @@ -16,7 +16,7 @@ VORTEX_DB_IMAGE= VORTEX_DB_IMAGE_BASE= # Drupal admin email. May need to be reset if database was sanitized. -DRUPAL_ADMIN_EMAIL="webmaster@your-site-url.example" +DRUPAL_ADMIN_EMAIL="webmaster@your-site-domain.example" # Password replacement used for sanitised database. VORTEX_PROVISION_SANITIZE_DB_PASSWORD="" @@ -42,7 +42,7 @@ DRUPAL_SITE_NAME="${VORTEX_PROJECT}" # Drupal site email. # Used only when installing from profile. -DRUPAL_SITE_EMAIL="webmaster@your-site-url.example" +DRUPAL_SITE_EMAIL="webmaster@your-site-domain.example" # Print output from Composer install. VORTEX_COMPOSER_VERBOSE=1 diff --git a/.vortex/docs/.utils/variables/extra/ci.variables.sh b/.vortex/docs/.utils/variables/extra/ci.variables.sh index 1edce6099..bdd5336e1 100755 --- a/.vortex/docs/.utils/variables/extra/ci.variables.sh +++ b/.vortex/docs/.utils/variables/extra/ci.variables.sh @@ -65,4 +65,4 @@ RENOVATE_DEPENDENCY_DASHBOARD=false RENOVATE_DRY_RUN=false # Commit author for self-hosted Renovate bot. -RENOVATE_GIT_AUTHOR='Renovate Self Hosted ' +RENOVATE_GIT_AUTHOR='Renovate Self Hosted ' diff --git a/.vortex/docs/content/workflows/variables.mdx b/.vortex/docs/content/workflows/variables.mdx index 9442d6b59..e5bcf61ed 100644 --- a/.vortex/docs/content/workflows/variables.mdx +++ b/.vortex/docs/content/workflows/variables.mdx @@ -88,7 +88,7 @@ Defined in: `docker-compose.yml`, `scripts/vortex/info.sh` Drupal admin email. May need to be reset if database was sanitized. -Default value: `webmaster@your-site-url.example` +Default value: `webmaster@your-site-domain.example` Defined in: `.env` @@ -172,7 +172,7 @@ Defined in: `.env` Drupal site email.
Used only when installing from profile. -Default value: `webmaster@your-site-url.example` +Default value: `webmaster@your-site-domain.example` Defined in: `.env`, `scripts/vortex/provision.sh` @@ -190,7 +190,7 @@ Stage file proxy origin. If using Shield, the HTTP authentication credentials will be automatically
added to the origin URL. -Default value: `https://www.your-site-url.example/` +Default value: `https://www.your-site-domain.example/` Defined in: `.env` @@ -274,7 +274,7 @@ Defined in: `CI config` Commit author for self-hosted Renovate bot. -Default value: `'Renovate Self Hosted '` +Default value: `'Renovate Self Hosted '` Defined in: `CI config` @@ -1242,7 +1242,7 @@ Defined in: `scripts/vortex/notify-email.sh` Email to send notifications from. -Default value: `webmaster@your-site-url.example` +Default value: `webmaster@your-site-domain.example` Defined in: `.env`, `scripts/vortex/notify-email.sh` @@ -1260,7 +1260,7 @@ Email address(es) to send notifications to. Multiple names can be specified as a comma-separated list of email addresses
with optional names in the format "email|name".
Example: "to1@example.com|Jane Doe, to2@example.com|John Doe" -Default value: `webmaster@your-site-url.example` +Default value: `webmaster@your-site-domain.example` Defined in: `.env`, `scripts/vortex/notify-email.sh` @@ -1580,7 +1580,7 @@ Sanitization email pattern. Applied if database sanitization is enabled.
@see https://vortex.drevops.com/workflows/build#sanitization -Default value: `user_%uid@your-site-url.example` +Default value: `user_%uid@your-site-domain.example` Defined in: `.env`, `scripts/vortex/provision-sanitize-db.sh` diff --git a/.vortex/installer/src/Command/InstallCommand.php b/.vortex/installer/src/Command/InstallCommand.php index 928e8cd8e..e2225942a 100644 --- a/.vortex/installer/src/Command/InstallCommand.php +++ b/.vortex/installer/src/Command/InstallCommand.php @@ -254,7 +254,7 @@ protected function collectAnswers(): void { $this->askForAnswer('module_prefix', 'What is your project-specific module prefix?'); $this->askForAnswer('profile', 'What is your custom profile machine name (leave empty to use "standard" profile)?'); $this->askForAnswer('theme', 'What is your theme machine name?'); - $this->askForAnswer('url', 'What is your site public URL?'); + $this->askForAnswer('domain', 'What is your site public domain?'); $this->askForAnswer('webroot', 'Web root (web, docroot)?'); $this->askForAnswer('provision_use_profile', 'Do you want to install from profile (leave empty or "n" for using database?'); diff --git a/.vortex/installer/src/Traits/PromptsTrait.php b/.vortex/installer/src/Traits/PromptsTrait.php index 727eb419a..c3cfdbef5 100644 --- a/.vortex/installer/src/Traits/PromptsTrait.php +++ b/.vortex/installer/src/Traits/PromptsTrait.php @@ -50,7 +50,7 @@ protected function getDefaultValueTheme(): mixed { return $this->getAnswer('machine_name'); } - protected function getDefaultValueUrl(): string { + protected function getDefaultValueDomain(): string { $value = $this->getAnswer('machine_name'); $value = str_replace('_', '-', $value); @@ -543,35 +543,8 @@ protected function discoverValueTheme(): ?string { return str_replace(['.info.yml', '.info'], '', $name); } - protected function discoverValueUrl(): ?string { - $webroot = $this->getAnswer('webroot'); - - $origin = NULL; - $path = $this->config->getDstDir() . sprintf('/%s/sites/default/settings.php', $webroot); - - if (!is_readable($path)) { - return NULL; - } - - $contents = file_get_contents($path); - if (!$contents) { - return NULL; - } - - // Drupal 8 and 9. - if (preg_match('/\$config\s*\[\'stage_file_proxy.settings\'\]\s*\[\'origin\'\]\s*=\s*[\'"]([^\'"]+)[\'"];/', $contents, $matches)) { - $origin = $matches[1]; - } - // Drupal 7. - elseif (preg_match('/\$conf\s*\[\'stage_file_proxy_origin\'\]\s*=\s*[\'"]([^\'"]+)[\'"];/', $contents, $matches)) { - $origin = $matches[1]; - } - - if ($origin) { - $origin = parse_url($origin, PHP_URL_HOST); - } - - return empty($origin) ? NULL : $origin; + protected function discoverValueDomain(): ?string { + return $this->getValueFromDstDotenv('DRUPAL_STAGE_FILE_PROXY_ORIGIN'); } protected function discoverValueWebroot(): ?string { @@ -724,10 +697,13 @@ protected function normaliseAnswerTheme(string $value): string { return Converter::toMachineName($value); } - protected function normaliseAnswerUrl(string $url): string { - $url = trim($url); + protected function normaliseAnswerDomain(string $value): string { + $value = trim($value); + $value = rtrim($value, '/'); + $value = str_replace([' ', '_'], '-', $value); + $value = preg_replace('/^https?:\/\//', '', $value); - return str_replace([' ', '_'], '-', $url); + return preg_replace('/^www\./', '', $value); } protected function normaliseAnswerWebroot(string $value): string { @@ -910,43 +886,38 @@ protected function processStringTokens(string $dir): void { $module_prefix_uppercase = strtoupper($module_prefix_camel_cased); $theme_camel_cased = Converter::toCamelCase($this->getAnswer('theme'), TRUE); $vortex_version_urlencoded = str_replace('-', '--', (string) $this->config->get('VORTEX_VERSION')); - $url = $this->getAnswer('url'); - $host = parse_url($url, PHP_URL_HOST); - $domain = $host ?: $url; - $domain_non_www = str_starts_with((string) $domain, "www.") ? substr((string) $domain, 4) : $domain; $webroot = $this->getAnswer('webroot'); // @formatter:off // phpcs:disable Generic.Functions.FunctionCallArgumentSpacing.TooMuchSpaceAfterComma // phpcs:disable Drupal.WhiteSpace.Comma.TooManySpaces - File::dirReplaceContent('your_site_theme', $this->getAnswer('theme'), $dir); - File::dirReplaceContent('YourSiteTheme', $theme_camel_cased, $dir); - File::dirReplaceContent('your_org', $this->getAnswer('org_machine_name'), $dir); - File::dirReplaceContent('YOURORG', $this->getAnswer('org'), $dir); - File::dirReplaceContent('www.your-site-url.example', $domain, $dir); - File::dirReplaceContent('your-site-url.example', $domain_non_www, $dir); - File::dirReplaceContent('ys_core', $this->getAnswer('module_prefix') . '_core', $dir . sprintf('/%s/modules/custom', $webroot)); - File::dirReplaceContent('ys_search', $this->getAnswer('module_prefix') . '_search', $dir . sprintf('/%s/modules/custom', $webroot)); - File::dirReplaceContent('ys_core', $this->getAnswer('module_prefix') . '_core', $dir . sprintf('/%s/themes/custom', $webroot)); - File::dirReplaceContent('ys_core', $this->getAnswer('module_prefix') . '_core', $dir . '/scripts/custom'); - File::dirReplaceContent('ys_search', $this->getAnswer('module_prefix') . '_search', $dir . '/scripts/custom'); - File::dirReplaceContent('YsCore', $module_prefix_camel_cased . 'Core', $dir . sprintf('/%s/modules/custom', $webroot)); - File::dirReplaceContent('YsSearch', $module_prefix_camel_cased . 'Search', $dir . sprintf('/%s/modules/custom', $webroot)); - File::dirReplaceContent('YSCODE', $module_prefix_uppercase, $dir); - File::dirReplaceContent('YSSEARCH', $module_prefix_uppercase, $dir); - File::dirReplaceContent('your-site', $machine_name_hyphenated, $dir); - File::dirReplaceContent('your_site', $this->getAnswer('machine_name'), $dir); - File::dirReplaceContent('YOURSITE', $this->getAnswer('name'), $dir); - File::dirReplaceContent('YourSite', $machine_name_camel_cased, $dir); - - File::replaceStringFilename('YourSiteTheme', $theme_camel_cased, $dir); - File::replaceStringFilename('your_site_theme', $this->getAnswer('theme'), $dir); - File::replaceStringFilename('YourSite', $machine_name_camel_cased, $dir); - File::replaceStringFilename('ys_core', $this->getAnswer('module_prefix') . '_core', $dir . sprintf('/%s/modules/custom', $webroot)); - File::replaceStringFilename('ys_search', $this->getAnswer('module_prefix') . '_search', $dir . sprintf('/%s/modules/custom', $webroot)); - File::replaceStringFilename('YsCore', $module_prefix_camel_cased . 'Core', $dir . sprintf('/%s/modules/custom', $webroot)); - File::replaceStringFilename('your_org', $this->getAnswer('org_machine_name'), $dir); - File::replaceStringFilename('your_site', $this->getAnswer('machine_name'), $dir); + File::dirReplaceContent('your_site_theme', $this->getAnswer('theme'), $dir); + File::dirReplaceContent('YourSiteTheme', $theme_camel_cased, $dir); + File::dirReplaceContent('your_org', $this->getAnswer('org_machine_name'), $dir); + File::dirReplaceContent('YOURORG', $this->getAnswer('org'), $dir); + File::dirReplaceContent('your-site-domain.example', $this->getAnswer('domain'), $dir); + File::dirReplaceContent('ys_core', $this->getAnswer('module_prefix') . '_core', $dir . sprintf('/%s/modules/custom', $webroot)); + File::dirReplaceContent('ys_search', $this->getAnswer('module_prefix') . '_search', $dir . sprintf('/%s/modules/custom', $webroot)); + File::dirReplaceContent('ys_core', $this->getAnswer('module_prefix') . '_core', $dir . sprintf('/%s/themes/custom', $webroot)); + File::dirReplaceContent('ys_core', $this->getAnswer('module_prefix') . '_core', $dir . '/scripts/custom'); + File::dirReplaceContent('ys_search', $this->getAnswer('module_prefix') . '_search', $dir . '/scripts/custom'); + File::dirReplaceContent('YsCore', $module_prefix_camel_cased . 'Core', $dir . sprintf('/%s/modules/custom', $webroot)); + File::dirReplaceContent('YsSearch', $module_prefix_camel_cased . 'Search', $dir . sprintf('/%s/modules/custom', $webroot)); + File::dirReplaceContent('YSCODE', $module_prefix_uppercase, $dir); + File::dirReplaceContent('YSSEARCH', $module_prefix_uppercase, $dir); + File::dirReplaceContent('your-site', $machine_name_hyphenated, $dir); + File::dirReplaceContent('your_site', $this->getAnswer('machine_name'), $dir); + File::dirReplaceContent('YOURSITE', $this->getAnswer('name'), $dir); + File::dirReplaceContent('YourSite', $machine_name_camel_cased, $dir); + + File::replaceStringFilename('YourSiteTheme', $theme_camel_cased, $dir); + File::replaceStringFilename('your_site_theme', $this->getAnswer('theme'), $dir); + File::replaceStringFilename('YourSite', $machine_name_camel_cased, $dir); + File::replaceStringFilename('ys_core', $this->getAnswer('module_prefix') . '_core', $dir . sprintf('/%s/modules/custom', $webroot)); + File::replaceStringFilename('ys_search', $this->getAnswer('module_prefix') . '_search', $dir . sprintf('/%s/modules/custom', $webroot)); + File::replaceStringFilename('YsCore', $module_prefix_camel_cased . 'Core', $dir . sprintf('/%s/modules/custom', $webroot)); + File::replaceStringFilename('your_org', $this->getAnswer('org_machine_name'), $dir); + File::replaceStringFilename('your_site', $this->getAnswer('machine_name'), $dir); File::dirReplaceContent('VORTEX_VERSION_URLENCODED', $vortex_version_urlencoded, $dir); File::dirReplaceContent('VORTEX_VERSION', $this->config->get('VORTEX_VERSION'), $dir); diff --git a/.vortex/installer/src/Traits/TuiTrait.php b/.vortex/installer/src/Traits/TuiTrait.php index 4724e5e98..88437c9ca 100644 --- a/.vortex/installer/src/Traits/TuiTrait.php +++ b/.vortex/installer/src/Traits/TuiTrait.php @@ -178,7 +178,7 @@ protected function printSummary(): void { $values['Module prefix'] = $this->getAnswer('module_prefix'); $values['Profile'] = $this->getAnswer('profile'); $values['Theme name'] = $this->getAnswer('theme'); - $values['URL'] = $this->getAnswer('url'); + $values['Domain'] = $this->getAnswer('domain'); $values['Web root'] = $this->getAnswer('webroot'); $values['Install from profile'] = $this->formatYesNo($this->getAnswer('provision_use_profile')); diff --git a/.vortex/tests/bats/_helper.bash b/.vortex/tests/bats/_helper.bash index 9462aba07..a794838f8 100644 --- a/.vortex/tests/bats/_helper.bash +++ b/.vortex/tests/bats/_helper.bash @@ -541,7 +541,7 @@ assert_files_present_drupal() { assert_dir_not_contains_string "${dir}" "your_site_theme" assert_dir_not_contains_string "${dir}" "your_org" assert_dir_not_contains_string "${dir}" "YOURORG" - assert_dir_not_contains_string "${dir}" "www.your-site-url.example" + assert_dir_not_contains_string "${dir}" "www.your-site-domain.example" # Assert all special comments were removed. assert_dir_not_contains_string "${dir}" "#;" assert_dir_not_contains_string "${dir}" "#;<" diff --git a/.vortex/tests/bats/install.parameters.bats b/.vortex/tests/bats/install.parameters.bats index 9bec299bf..6b738c662 100644 --- a/.vortex/tests/bats/install.parameters.bats +++ b/.vortex/tests/bats/install.parameters.bats @@ -64,7 +64,7 @@ load _helper.bash assert_output_contains " Module prefix: sw " assert_output_contains " Profile: standard " assert_output_contains " Theme name: star_wars " - assert_output_contains " URL: star-wars.com " + assert_output_contains " Domain: star-wars.com " assert_output_contains " Web root: web " assert_output_contains " Install from profile: No " assert_output_contains " Database download source: curl " @@ -116,7 +116,7 @@ load _helper.bash assert_output_contains " Module prefix: sw " assert_output_contains " Profile: standard " assert_output_contains " Theme name: star_wars " - assert_output_contains " URL: star-wars.com " + assert_output_contains " Domain: star-wars.com " assert_output_contains " Web root: web " assert_output_contains " Install from profile: No " assert_output_contains " Database download source: curl " @@ -170,7 +170,7 @@ load _helper.bash assert_output_contains " Module prefix: s_w " assert_output_contains " Profile: s_w_profile " assert_output_contains " Theme name: light_saber " - assert_output_contains " URL: resistance-forever.com " + assert_output_contains " Domain: resistance-forever.com " assert_output_contains " Web root: web " assert_output_contains " Install from profile: No " assert_output_contains " Database download source: curl " @@ -223,7 +223,7 @@ load _helper.bash assert_output_contains " Module prefix: s_w " assert_output_contains " Profile: s_w_profile " assert_output_contains " Theme name: light_saber " - assert_output_contains " URL: resistance-forever.com " + assert_output_contains " Domain: resistance-forever.com " assert_output_contains " Web root: rootdoc " assert_output_contains " Install from profile: No " assert_output_contains " Database download source: curl " @@ -256,7 +256,7 @@ load _helper.bash assert_output_contains " Profile: standard " assert_output_contains " Install from profile: No " assert_output_contains " Theme name: resisting " - assert_output_contains " URL: www.resistance-star-wars.com " + assert_output_contains " Domain: resistance-star-wars.com " assert_output_contains " Web root: web " assert_output_contains " Database download source: curl " assert_output_contains " Database store type: file " @@ -311,7 +311,7 @@ load _helper.bash assert_output_contains " Module prefix: another_resist " assert_output_contains " Profile: standard " assert_output_contains " Theme name: resisting " - assert_output_contains " URL: www.resistance-star-wars.com " + assert_output_contains " Domain: resistance-star-wars.com " assert_output_contains " Web root: web " assert_output_contains " Install from profile: No " assert_output_contains " Database download source: curl " @@ -373,7 +373,7 @@ load _helper.bash assert_output_contains " Module prefix: w_s " assert_output_contains " Profile: s_w_profile " assert_output_contains " Theme name: light_saber " - assert_output_contains " URL: resistance-forever.com " + assert_output_contains " Domain: resistance-forever.com " assert_output_contains " Web root: web " assert_output_contains " Install from profile: No " assert_output_contains " Database download source: container_registry " @@ -415,10 +415,6 @@ fixture_preinstalled() { mktouch "${webroot}/sites/all/themes/custom/resisting/resisting.info.yml" mktouch "${webroot}/sites/all/themes/custom/yetanothertheme/yetanothertheme.info.yml" - # Sets 'url' to 'www.resistance-star-wars.com'. - mkdir -p "${webroot}/sites/default" - echo " \$config['stage_file_proxy.settings']['origin'] = 'http://www.resistance-star-wars.com/';" >"${webroot}/sites/default/settings.php" - # Sets 'ci_provider' to 'GitHub Actions'. mktouch ".github/workflows/build-test-deploy.yml" @@ -431,6 +427,9 @@ fixture_preinstalled() { echo "VORTEX_WEBROOT=${webroot}" >>.env + # Sets 'domain' to 'www.resistance-star-wars.com'. + echo "DRUPAL_STAGE_FILE_PROXY_ORIGIN=http://www.resistance-star-wars.com/" >>.env + # Sets 'fresh_install' to 'No'. echo "VORTEX_PROVISION_USE_PROFILE=0" >>.env diff --git a/hooks/library/domains.txt b/hooks/library/domains.txt index aa6771663..0e31e9f2e 100644 --- a/hooks/library/domains.txt +++ b/hooks/library/domains.txt @@ -1 +1 @@ -$TARGET_ENV_REMAP.your-site-url.example +$TARGET_ENV_REMAP.your-site-domain.example