Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[WIP] Create Install Command #118

Draft
wants to merge 40 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 13 commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
0cb1cae
Compile Assets
AlexJump24 Oct 13, 2024
3a983bb
Compile Assets
AlexJump24 Oct 13, 2024
b2f527a
Initial work on the install command to save app settings
AlexJump24 Oct 13, 2024
c21b990
Minor tweaks
AlexJump24 Oct 14, 2024
299e1d9
Add config item ready to set up configuring own database connection f…
AlexJump24 Oct 14, 2024
c32bc4f
Move items out of testbech to install command itself.
AlexJump24 Oct 14, 2024
2fc0720
move migrations higher in chain at least for now
AlexJump24 Oct 14, 2024
a42fea2
Compile Assets
AlexJump24 Oct 14, 2024
760bd43
update
AlexJump24 Oct 15, 2024
c08c450
Write to env file if prompted for configuration
AlexJump24 Oct 16, 2024
a3aeab4
tidy up
AlexJump24 Oct 17, 2024
44c3478
changes from feedback
AlexJump24 Oct 17, 2024
6b153a1
fix pipeline issue
AlexJump24 Oct 17, 2024
0305ec8
Pipeline before passed oddly so not needed
AlexJump24 Oct 18, 2024
44bb282
feedback + tidy up
AlexJump24 Oct 18, 2024
f001b7a
Compile Assets
AlexJump24 Oct 13, 2024
0971029
Initial work on the install command to save app settings
AlexJump24 Oct 13, 2024
fe389a3
Minor tweaks
AlexJump24 Oct 14, 2024
67769aa
Add config item ready to set up configuring own database connection f…
AlexJump24 Oct 14, 2024
419249e
Move items out of testbech to install command itself.
AlexJump24 Oct 14, 2024
612e94e
move migrations higher in chain at least for now
AlexJump24 Oct 14, 2024
5f87f58
Compile Assets
AlexJump24 Oct 14, 2024
9d2411b
update
AlexJump24 Oct 15, 2024
d5c85b4
Write to env file if prompted for configuration
AlexJump24 Oct 16, 2024
4331b0d
tidy up
AlexJump24 Oct 17, 2024
c8baa93
changes from feedback
AlexJump24 Oct 17, 2024
9d7bc2f
fix pipeline issue
AlexJump24 Oct 17, 2024
da9f994
Pipeline before passed oddly so not needed
AlexJump24 Oct 18, 2024
c628bc5
feedback + tidy up
AlexJump24 Oct 18, 2024
fd95eba
Merge branch 'main' into feature/issue-104-install-command
jbrooksuk Nov 30, 2024
e84e1c7
Merge branch 'feature/issue-104-install-command' of https://github.co…
AlexJump24 Dec 27, 2024
3aea0e0
Merge branch 'main' into feature/issue-104-install-command
AlexJump24 Dec 27, 2024
7aecf3e
Merge branch 'main' of https://github.com/AlexJump24/core
AlexJump24 Dec 27, 2024
fccf60f
Merge branch 'main' into feature/issue-104-install-command
AlexJump24 Dec 27, 2024
af31e0a
Add optional create user command
AlexJump24 Dec 27, 2024
790926c
change default
AlexJump24 Dec 27, 2024
4ff5de2
fix stan issues
AlexJump24 Dec 27, 2024
4184a49
Prevent user table being truncated on install in another Laravel project
AlexJump24 Dec 29, 2024
470e982
Change order of things to prevent issues with database connection
AlexJump24 Dec 29, 2024
1c0c332
Add trait to models to overload connection name to use config value f…
AlexJump24 Jan 1, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@
"illuminate/events": "^11.0",
"illuminate/queue": "^11.0",
"illuminate/support": "^11.0",
"laravel/prompts": "^0.1",
"nesbot/carbon": "^2.70",
"spatie/laravel-query-builder": "^5.5",
"spatie/laravel-settings": "^3.2",
Expand Down
10 changes: 10 additions & 0 deletions config/cachet.php
Original file line number Diff line number Diff line change
Expand Up @@ -107,4 +107,14 @@
|
*/
'docker' => env('CACHET_DOCKER', false),

/*
|--------------------------------------------------------------------------
| Cachet Database Connection
|--------------------------------------------------------------------------
|
| Support using an alternative database connection, defaults to default connecton of application
|
*/
'database_connection' => env('CACHET_DB_CONNECTION', 'default'),
];
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,15 @@ public function up(): void
rescue(fn () => $this->migrator->add('app.install_id', Str::random(40)));
rescue(fn () => $this->migrator->add('app.name', 'Cachet'));
rescue(fn () => $this->migrator->add('app.domain'));
rescue(fn () => $this->migrator->add('app.about'));
rescue(fn () => $this->migrator->add('app.about', <<<'ABOUT'
Cachet is a **beautiful** and **powerful** open-source status page system.

To access the [dashboard](/dashboard), use the following credentials:
- `[email protected]`
- `test123`

Please [consider sponsoring](https://github.com/cachethq/cachet?sponsor=1) the continued development of Cachet.
ABOUT));
rescue(fn () => $this->migrator->add('app.timezone', 'UTC'));
rescue(fn () => $this->migrator->add('app.locale', 'en'));
rescue(fn () => $this->migrator->add('app.incident_days', 7));
Expand Down
21 changes: 0 additions & 21 deletions database/seeders/DatabaseSeeder.php
Original file line number Diff line number Diff line change
Expand Up @@ -161,27 +161,6 @@ public function run(): void
'engine' => IncidentTemplateEngineEnum::twig,
]);

$appSettings = app(AppSettings::class);
$appSettings->name = 'Cachet v3.x Demo';
$appSettings->about = <<<'ABOUT'
Cachet is a **beautiful** and **powerful** open-source status page system.

To access the [dashboard](/dashboard), use the following credentials:
- `[email protected]`
- `test123`

Please [consider sponsoring](https://github.com/cachethq/cachet?sponsor=1) the continued development of Cachet.
ABOUT;
$appSettings->show_support = true;
$appSettings->timezone = 'UTC';
$appSettings->show_timezone = false;
$appSettings->only_disrupted_days = false;
$appSettings->incident_days = 7;
$appSettings->refresh_rate = null;
$appSettings->dashboard_login_link = true;
$appSettings->major_outage_threshold = 25;
$appSettings->save();

$customizationSettings = app(CustomizationSettings::class);
$customizationSettings->header = <<<'HTML'
<script src="https://cdn.usefathom.com/script.js" data-site="NQKCLYJJ" defer></script>
Expand Down
1 change: 0 additions & 1 deletion public/build/assets/cachet-BIHPJ10n.css

This file was deleted.

18 changes: 0 additions & 18 deletions public/build/assets/cachet-DCZQ8JcZ.js

This file was deleted.

1 change: 1 addition & 0 deletions public/build/assets/cachet.30cb9e62.css

Large diffs are not rendered by default.

18 changes: 18 additions & 0 deletions public/build/assets/cachet.6122e927.js

Large diffs are not rendered by default.

1 change: 0 additions & 1 deletion public/build/assets/theme-BbibcaDc.css

This file was deleted.

1 change: 1 addition & 0 deletions public/build/assets/theme.8251fad4.css

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion public/build/manifest.json
Original file line number Diff line number Diff line change
Expand Up @@ -15,4 +15,4 @@
"src": "resources/js/cachet.js",
"isEntry": true
}
}
}
1 change: 1 addition & 0 deletions src/CachetCoreServiceProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -151,6 +151,7 @@ private function registerCommands(): void
{
if ($this->app->runningInConsole()) {
$this->commands([
Commands\InstallCommand::class,
Commands\SendBeaconCommand::class,
Commands\VersionCommand::class,
]);
Expand Down
144 changes: 144 additions & 0 deletions src/Commands/InstallCommand.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,144 @@
<?php

namespace Cachet\Commands;

use Cachet\Database\Seeders\DatabaseSeeder;
use Cachet\Settings\AppSettings;
use Cachet\Settings\Attributes\Description;
use Illuminate\Console\Command;
use Illuminate\Support\Facades\File;
use Illuminate\Support\Sleep;
use Illuminate\Support\Str;
use Laravel\Prompts\Prompt;
use ReflectionClass;
use ReflectionProperty;
use function Laravel\Prompts\confirm;
use function Laravel\Prompts\info;
use function Laravel\Prompts\intro;
use function Laravel\Prompts\text;


class InstallCommand extends Command
{
protected $name = 'cachet:install';

protected $description = 'Install Cachet';

public function handle(AppSettings $settings)
{
intro('Welcome to the Cachet installer!');

Sleep::for(2)->seconds();

$this->call('migrate', ['--seed' => true, '--seeder' => DatabaseSeeder::class]);

if (confirm('Do you want to configure Cachet before installing?', true)) {
info('Configuring Cachet...');
$this->configureEnvironmentSettings();
$settings = $this->configureDatabaseSettings($settings);
}

info('Installing Cachet...');

$this->call('filament:assets');

$settings->save();

info('Cachet is installed ⚡');

return Command::SUCCESS;
}

protected function configureEnvironmentSettings(): void
{
$path = text(
'Which path do you want Cachet to be accessible from?',
default: config('cachet.path')
);

$title = text(
'What will the title of your status page be?',
default: config('cachet.title')
);

$connection = text(
'Which database connection do you wish to use for Cachet?',
default: config('cachet.database_connection')
);

$beacon = confirm(
'Do you wish to send anonymous data to cachet to help us understand how Cachet is used?',
default: config('cachet.beacon')
);

$this->writeEnv([
'CACHET_PATH' => $path,
'CACHET_TITLE' => $title,
'CACHET_DB_CONNECTION' => $connection,
'CACHET_BEACON' => $beacon,
]);
}

protected function writeEnv(array $values): void
{
$environmentFile = app()->environmentFile();
$environmentPath = app()->environmentPath();
$fullPath = $environmentPath . '/' . $environmentFile;

$envFileContents = File::lines($fullPath)->collect();

foreach ($values as $key => $value) {
$existingKey = $envFileContents->search(function ($line) use ($key) {
return Str::contains($line, $key, true);
});

$value = match (true) {
Str::contains($value, ' ') => Str::wrap($value,'"'),
$value === true => 'true',
$value === false => 'false',
default => $value
};

if ($existingKey === false) {
$envFileContents->push($key . '=' . $value);
} else {
$envFileContents->put($existingKey, $key . '=' . $value);
}
}

File::put($fullPath, $envFileContents->implode("\n"));
}

protected function configureDatabaseSettings(AppSettings $settings): AppSettings
{
collect(
(new ReflectionClass($settings))->getProperties(ReflectionProperty::IS_PUBLIC)
)
->filter(fn (ReflectionProperty $property) => array_key_exists($property->getName(), $settings->installable()) )
->each(function (ReflectionProperty $property) use ($settings) {
$descriptionAttribute = $property->getAttributes(Description::class);

if (empty($descriptionAttribute)) {
return;
}

$descriptionAttributeClass = $descriptionAttribute[0]->newInstance();
$default = $descriptionAttributeClass->getDefault();
$required = $descriptionAttributeClass->getRequired();
jbrooksuk marked this conversation as resolved.
Show resolved Hide resolved

if ($required === false) {
return;
}

$value = match($property->getType()?->getName()) {
'bool' => confirm($default ?? $property->getName()),
default => text($default ?? $property->getName(), default: $property->getDefaultValue() ?? '', required: true),
};

$settings->{$property->getName()} = $value;
})
->pluck('name');

return $settings;
}
}
20 changes: 20 additions & 0 deletions src/Settings/AppSettings.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,34 +2,54 @@

namespace Cachet\Settings;

use Cachet\Settings\Attributes\Description;
use Illuminate\Support\Arr;
use Spatie\LaravelSettings\Settings;

class AppSettings extends Settings
{
#[Description('The unique install ID of the application for telemetry.')]
public string $install_id;

#[Description('What is the name of your application?')]
public ?string $name = 'Cachet';

#[Description('What is your application about?', required: false)]
public ?string $about;

#[Description('Do you want to show support for Cachet?')]
public bool $show_support = true;

#[Description('What timezone is is the application located in?')]
public string $timezone = 'UTC';

#[Description('Would you like to show your timezone on the status page?')]
public bool $show_timezone = false;

#[Description('Would you like to only show the days with disruption?')]
public bool $only_disrupted_days = false;

#[Description('How many incident days should be shown in the timeline?')]
public int $incident_days = 7;

#[Description('After how many seconds should the status page automatically refresh?', required: false)]
public ?int $refresh_rate;

#[Description('Should the dashboard login link be shown?')]
public bool $dashboard_login_link = true;

#[Description('Major outage threshold %')]
public int $major_outage_threshold = 25;

public static function group(): string
{
return 'app';
}

public function installable(): array
{
return Arr::except(get_class_vars(__CLASS__), [
'install_id',
]);
}
}
21 changes: 21 additions & 0 deletions src/Settings/Attributes/Description.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
<?php

namespace Cachet\Settings\Attributes;

use Attribute;

#[Attribute(Attribute::TARGET_PROPERTY)]
final class Description
{
public function __construct(private readonly string $default, private readonly bool $required = true) {}

public function getDefault(): string
jbrooksuk marked this conversation as resolved.
Show resolved Hide resolved
{
return $this->default;
}

public function getRequired(): bool
jbrooksuk marked this conversation as resolved.
Show resolved Hide resolved
{
return $this->required;
}
}
5 changes: 1 addition & 4 deletions testbench.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -18,12 +18,9 @@ workbench:
to: public/vendor/cachethq/cachet
build:
- asset-publish
- filament:assets
- create-sqlite-db
- db:wipe
- migrate:refresh:
--seed: true
--seeder: Cachet\Database\Seeders\DatabaseSeeder
- cachet:install
assets:
- query-builder-config
- cachet-assets
Expand Down
61 changes: 61 additions & 0 deletions tests/Unit/Commands/InstallCommandTest.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
<?php

namespace Tests\Unit\Commands;

use Cachet\Settings\AppSettings;
use Illuminate\Support\Facades\File;

it('runs install command successfully without configuration', function () {
$this->artisan('cachet:install')
->expectsOutputToContain('Welcome to the Cachet installer!')
->expectsConfirmation('Do you want to configure Cachet before installing?', 'no')
->expectsOutputToContain('Installing Cachet...')
->expectsOutputToContain('Cachet is installed ⚡')
->assertSuccessful();
});

it('updates app settings and config filewhen configuration is passed', function () {
File::copy(base_path('.env.example'), base_path('.env'));

$this->artisan('cachet:install')
->expectsOutputToContain('Welcome to the Cachet installer!')
->expectsConfirmation('Do you want to configure Cachet before installing?', 'yes')
->expectsOutputToContain('Configuring Cachet...')
->expectsQuestion('Which path do you want Cachet to be accessible from?', '/status')
->expectsQuestion('What will the title of your status page be?', 'Laravel Envoyer')
->expectsQuestion('Which database connection do you wish to use for Cachet?', 'default')
->expectsQuestion('Do you wish to send anonymous data to cachet to help us understand how Cachet is used?', true)
->expectsQuestion('What is the name of your application?', 'Laravel Envoyer')
->expectsConfirmation('Do you want to show support for Cachet?', 'yes')
->expectsQuestion('What timezone is is the application located in?', 'America/New_York')
->expectsConfirmation('Would you like to show your timezone on the status page?', 'yes')
->expectsConfirmation('Would you like to only show the days with disruption?', 'yes')
->expectsQuestion('How many incident days should be shown in the timeline?', 14)
->expectsConfirmation('Should the dashboard login link be shown?', 'no')
->expectsQuestion('Major outage threshold %', 50)
->expectsOutputToContain('Installing Cachet...')
->expectsOutputToContain('Cachet is installed ⚡')
->assertSuccessful();

$envContent = file_get_contents(base_path('.env'));

expect($envContent)
->toContain('CACHET_PATH=/status')
->toContain('CACHET_TITLE="Laravel Envoyer"')
->toContain('CACHET_DB_CONNECTION=default')
->toContain('CACHET_BEACON=true');

$settings = app(AppSettings::class);
expect($settings->name)->toBe('Laravel Envoyer')
->and($settings->show_support)->toBeTrue()
->and($settings->timezone)->toBe('America/New_York')
->and($settings->show_timezone)->toBeTrue()
->and($settings->only_disrupted_days)->toBeTrue()
->and($settings->incident_days)->toBe(14)
->and($settings->dashboard_login_link)->toBeFalse()
->and($settings->major_outage_threshold)->toBe(50);
});

afterAll(function() {
File::delete(base_path('.env'));
});