cleaning up before publishing

This commit is contained in:
Boudewijn 2025-03-03 22:10:40 +01:00
parent 934dfddcb3
commit 3988f71f09
13 changed files with 48 additions and 312 deletions

View file

@ -1,4 +1 @@
File containing the license of your package.
More information here:
https://choosealicense.com/
AGPL-3

View file

@ -1,63 +0,0 @@
<?php
namespace App\Console\Commands;
use App\Models\User;
use Illuminate\Console\Command;
use Illuminate\Support\Facades\Validator;
class CreateUser extends Command
{
/**
* The name and signature of the console command.
*
* @var string
*/
protected $signature = 'app:create-user
{name : The username of the user}
{email : The email of the user}
{password : Password in plain text (will be encrypted automatically)}
{--unverified : Keep the user unverified. Otherwise, email verified will be set at current time.}';
/**
* The console command description.
*
* @var string
*/
protected $description = 'Create a user to access the admin panel';
private array $rules = [
'name' => ['required', 'min:4', 'unique:users,name'],
'email' => ['required', 'email', 'unique:users,email'],
'password' => ['required', 'min:6'],
];
/**
* Execute the console command.
*/
public function handle(): int
{
$validator = Validator::make([
'name' => $this->argument('name'),
'email' => $this->argument('email'),
'password' => $this->argument('password'),
], $this->rules);
try {
$userData = $validator->validated();
$userData['password'] = bcrypt($userData['password']);
$userData['email_verified_at'] = $this->option('unverified') ? NULL : now();
User::factory()->create($userData);
$this->info('User created.');
} catch (\Exception $e) {
$this->error("ERROR: {$e->getMessage()} [Code: {$e->getCode()}]");
return 1;
}
return 0;
}
}

View file

@ -1,103 +0,0 @@
<?php
namespace Tests\Feature\Console;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Illuminate\Foundation\Testing\WithFaker;
use Illuminate\Support\Carbon;
use Tests\TestCase;
class CreateUserTest extends TestCase
{
use RefreshDatabase;
public function test_user_creates_with_valid_args(): void
{
$this->artisan('app:create-user', [
'name' => 'ajaxray',
'email' => 'user@ajaxray.com',
'password' => '12345678',
])->expectsOutputToContain('User created');
$this->assertDatabaseCount('users', 1);
}
public function test_cannot_create_user_with_invalid_email(): void
{
$this->artisan('app:create-user', [
'name' => 'ajaxray',
'email' => 'user@.com',
'password' => '222222',
])->assertFailed()
->expectsOutputToContain('must be a valid email address.');
$this->assertDatabaseEmpty('users');
}
public function test_cannot_create_user_with_duplicate_name(): void
{
$this->artisan('app:create-user', [
'name' => 'ajaxray',
'email' => 'user1@ajaxray.com',
'password' => '111111',
])->assertSuccessful();
$this->artisan('app:create-user', [
'name' => 'ajaxray',
'email' => 'user2@ajaxray.com',
'password' => '222222',
])->assertFailed()
->expectsOutputToContain('name has already been taken.');
$this->assertDatabaseCount('users', 1);
}
public function test_cannot_create_user_with_duplicate_email(): void
{
$this->artisan('app:create-user', [
'name' => 'user1',
'email' => 'user@ajaxray.com',
'password' => '111111',
])->assertSuccessful();
$this->artisan('app:create-user', [
'name' => 'user2',
'email' => 'user@ajaxray.com',
'password' => '222222',
])->assertFailed()
->expectsOutputToContain('email has already been taken.');
$this->assertDatabaseCount('users', 1);
}
public function test_created_users_are_verified_by_default(): void
{
$this->assertDatabaseEmpty('users');
$this->freezeTime(function (Carbon $time) {
$dateTimeStr = date('Y-m-d H:i:s');
$this->artisan('app:create-user', [
'name' => 'user1',
'email' => 'user@ajaxray.com',
'password' => '111111',
])->assertSuccessful();
$this->assertDatabaseHas('users', ['email_verified_at' => $dateTimeStr]);
});
}
public function test_created_users_can_be_kept_unverified(): void
{
$this->assertDatabaseEmpty('users');
$this->artisan('app:create-user', [
'name' => 'user1',
'email' => 'user@ajaxray.com',
'password' => '111111',
'--unverified' => true
])->assertSuccessful();
$this->assertDatabaseHas('users', ['email_verified_at' => null]);
}
}

View file

@ -1,4 +1,7 @@
If you are the person in your family, group of friends or colleauges that manages the servers, domain names, online accounts and other hosted services, than this app is for you.
My Idlers is a useful register for keeping track of your servers and online hosting accounts.
It is a single-user app; if more than one person wants to use this functionality, it needs to be installed multiple times.
Besides that, for now, it needs a (sub)domain for itself.
@ -7,21 +10,13 @@ This is my first app packaging attempt; there may be some rough edges.
What works:
* installing the app and using its features
* uninstalling
What does not (yet?) work:
* no backup
* no explicit fail2ban config
* backup
* moving the app to another domain
What does not (yet?) work or is untested:
* installing the app in subfolder
* LDAP integration
What I intend to work on enabling:
* YNH backup integration
* create your account from this install page, instead of manually in the app on first use. Once that is done, I will change the default group from "all_users" to "visitors"
* moving app to another domain
* installing to subdirectory
* see whether fail2ban needs a specific config, add if needed
* restore from backup
* API access
![Screenshot of My Idlers](./doc/screenshots/my_idlers.jpg)

View file

@ -1 +1,24 @@
Ceci est une fausse description des fonctionalités de l'app
(machine translate from English)
Si vous êtes la personne de votre famille, de votre groupe d'amis ou de vos collègues qui gère les serveurs, les noms de domaine, les comptes en ligne et d'autres services hébergés, alors cette application est faite pour vous.
My Idlers est un registre utile pour suivre vos serveurs et vos comptes d'hébergement en ligne.
C'est une application monoposte; si plusieurs personnes veulent utiliser cette fonctionnalité, il faut l'installer plusieurs fois. De plus, pour l'instant, elle nécessite un (sous-)domaine dédié.
Il s'agit de ma première tentative d'emballage d'application; il peut y avoir quelques imperfections.
Ce qui fonctionne :
* installation de l'application et utilisation de ses fonctionnalités
* désinstallation
* sauvegarde
* déplacement de l'application vers un autre domaine
Ce qui ne fonctionne pas (encore) ou n'est pas testé :
* installation de l'application dans un sous-dossier
* intégration LDAP
* restauration à partir d'une sauvegarde
* accès API
![Screenshot of My Idlers](./doc/screenshots/my_idlers.jpg)

View file

@ -4,4 +4,4 @@ The app install dir is `__INSTALL_DIR__`
The app id is `__ID__`
Please open the app to create an account. The account will unfortunately not be synchronized with Yunohost users.
There was an attempt to create a user with the credentials you provided. Please open the app to log in. If presented with a "registration" page instead of "login", please create an account yourself. Either way, the account will unfortunately not be synchronized with Yunohost users.

View file

@ -1 +1,3 @@
Ceci est un faux disclaimer à présenter avant l'installation
Pour l'instant, seuls le domaine et le groupe peuvent être définis.
Après l'installation, visitez l'application pour créer un compte. Comme il s'agit d'une application monoposte, la page de création de compte ne sera plus disponible par la suite.

View file

@ -36,16 +36,14 @@ ram.runtime = "100M"
[install.domain]
type = "domain"
# unable to get this to work, disable for now
# [install.path]
# type = "path"
# default = "/idlers"
# unable to get this to work, disable for now
#[install.path]
#type = "path"
#default = "/idlers"
[install.init_main_permission]
type = "group"
default = "all_users"
# for now no credentials configured, make it install first
[install.user]
help.en = "The name of the user"
type = "string"
@ -56,19 +54,12 @@ ram.runtime = "100M"
help.en = "Use the help field to add an information for the admin about this question."
help.fr = "Utilisez le champ aide pour ajouter une information à l'intention de l'administrateur à propos de cette question."
type = "password"
# [install.token]
# type = "string"
[resources]
# See the packaging documentation for the full set
# of explanation regarding the behavior and properties for each of those
[resources.sources]
[resources.sources.main]
# This will pre-fetch the asset which can then be deployed during the install/upgrade scripts with :
# ynh_setup_source --dest_dir="$install_dir"
# You can also define other assets than "main" and add --source_id="foobar" in the previous command
# This will pre-fetch the asset
url = "https://github.com/cp6/my-idlers/archive/refs/tags/3.0.zip"
sha256 = "f92a6dc9c98ec1e3837939db571833d28e72b992f5a9611925e9d242fcb40f72"
@ -79,16 +70,12 @@ ram.runtime = "100M"
group = "www-data:r-x"
[resources.permissions]
# This will configure SSOwat permission for $domain/$path/
# The initial allowed group of user is configured via the init_main_permission question (public=visitors, private=all_users)
main.url = "/"
[resources.ports]
# This will pick a random port for reverse-proxying and store it as the $port setting
[resources.apt]
packages = "php8.3, php8.3-cli, php8.3-mbstring, php8.3-xml, php8.3-mysql, php8.3-intl, php8.3-pdo, php8.3-intl, composer, mariadb-server, mariadb-client, php8.3-curl"
[resources.database]
# This will automatically provision/deprovison a MySQL DB and store the corresponding credentials in settings $db_user, $db_name, $db_pwd
type = "mysql"

View file

@ -10,11 +10,6 @@ source /usr/share/yunohost/helpers
ynh_print_info "Declaring files to be backed up..."
### N.B. : the following 'ynh_backup' calls are only a *declaration* of what needs
### to be backuped and not an actual copy of any file. The actual backup that
### creates and fills the archive with the files happens in the core after this
### script is called. Hence ynh_backups calls take basically 0 seconds to run.
#=================================================
# BACKUP THE APP MAIN DIR
#=================================================
@ -25,21 +20,13 @@ ynh_backup "$install_dir"
# BACKUP SYSTEM CONFIGURATION
#=================================================
# Backup the NGINX configuration
ynh_backup "/etc/nginx/conf.d/$domain.d/$app.conf"
# Backup the Fail2Ban config
#ynh_backup "/etc/fail2ban/jail.d/$app.conf"
#ynh_backup "/etc/fail2ban/filter.d/$app.conf"
#=================================================
# BACKUP THE MYSQL DATABASE
#=================================================
ynh_print_info "Backing up the MySQL database..."
### (However, things like MySQL dumps *do* take some time to run, though the
### copy of the generated dump to the archive still happens later)
ynh_mysql_dump_db > db.sql
#=================================================

View file

@ -1,6 +1,5 @@
#!/bin/bash
## this script is only run if actual change to domain/path is detected, if you're here either $domain or $path changed
## new location is available via $domain and $path (or $new_domain and $new_path variables if you want to be explicit)
## old values are available via, you guessed it, $old_domain and $old_path
@ -15,7 +14,7 @@ source /usr/share/yunohost/helpers
# STOP SYSTEMD SERVICE
#=================================================
## My Idlers runs as a Laravel app on nginx; keep that running.
## My Idlers runs as a Laravel app on nginx; keep nginx running, so no stop
#=================================================
# MODIFY URL IN NGINX CONF
@ -33,7 +32,7 @@ ynh_config_change_url_nginx
sed -i -e 's/$old_domain/new_domain/g' $install_dir/.env
sed -i -e 's/$old_path/new_path/g' $install_dir/.env
## perhaps a better option, ynh_replace + ynh_store_file_checksum
## perhaps a better option than sed: ynh_replace + ynh_store_file_checksum; try it out later
# ynh_replace --match=$old_domain --replace="$new_domain --file="$install_dir/.env"
# ynh_replace --match=$old_path --replace="$new_path --file="$install_dir/.env"
ynh_store_file_checksum "$install_dir/.env"

View file

@ -7,25 +7,6 @@
source _common.sh
source /usr/share/yunohost/helpers
### Install parameters are automatically saved as settings
###
### Settings are automatically loaded as bash variables
### in every app script context, therefore typically these will exist:
### - $domain
### - $path
### - $language
### ... etc
###
### Resources defined in the manifest are provisioned prior to this script
### and corresponding settings are also available, such as:
### - $install_dir
### - $port
### - $db_name
### ...
###
### $app is the app id (i.e. 'example' for first install,
### or 'example__2', '__3'... for multi-instance installs)
#=================================================
# DOWNLOAD, CHECK AND UNPACK SOURCE
#=================================================
@ -44,48 +25,18 @@ ynh_config_add --template=.env --destination="$install_dir/.env"
chmod 640 "$install_dir/.env"
chown "$app:www-data" "$install_dir/.env"
# Inject CreateUser.php into app/Console/Commands/ to extend the upstream version with this functionality
# Using this, we can create the user with the credentials provided
#ynh_config_add --template=CreateUser.php --destination="$install_dir/app/Console/Commands/CreateUser.php"
#ynh_config_add --template=CreateUserTest.php --destination="$install_dir/app/Console/Commands/CreateUserTest.php"
# Scratch that. CreateUser crashes without the test file, the test file depends on class testcases. I'll look for a solution that does not pull in a boatload of dependencies.
#=================================================
# SYSTEM CONFIGURATION
#=================================================
ynh_script_progression "Adding system configurations related to $app..."
### `ynh_config_add_phpfpm` is used to set up a PHP config.
### You can remove it if your app doesn't use PHP.
### `ynh_config_add_phpfpm` will use the files conf/extra_php-fpm.conf
### If you're not using these lines:
### - You can remove these files in conf/.
### - Remove the section "BACKUP THE PHP-FPM CONFIGURATION" in the backup script
### - Remove also the section "REMOVE PHP-FPM CONFIGURATION" in the remove script
### - As well as the section "RESTORE THE PHP-FPM CONFIGURATION" in the restore script
### with the reload at the end of the script.
### - And the section "PHP-FPM CONFIGURATION" in the upgrade script
# Create a PHP-FPM config (with conf/extra_php-fpm.conf being appended to it)
# (added to nginx.conf)
#ynh_config_add_phpfpm
# Create a dedicated NGINX config using the conf/nginx.conf template
# (the PHP-FPM config is here as well)
ynh_config_add_nginx
# systemd : not applicable, runs via php-fpm and nginx
# Create a dedicated Fail2Ban config
# todo
#ynh_config_add_fail2ban --logpath="/var/log/nginx/${domain}-error.log" --failregex="Regex to match into the log for a failed login"
#=================================================
# INSTALL APP WITH COMPOSER
#=================================================
ynh_script_progression "Installing app with Composer..."
#wbk fixme manual installation needed an update before executing install
ynh_composer_install
ynh_composer_exec install --no-dev
@ -94,24 +45,16 @@ ynh_composer_exec install --no-dev
#=================================================
ynh_script_progression "configuring $app..."
# Generate a random alfanumeric string as API token
pushd "$install_dir"
"php$php_version" artisan key:generate -n --force --env
"php$php_version" artisan make:database $app
"php$php_version" artisan migrate:fresh --seed -n --force
#"php$php_version" artisan app:create-user "$user" "$email" "$password"
# Alternative method: direct injection of credentials using Artisan Tinker:
echo "use App\Models\User; User::create(['name' => '$user', 'email' => '$email' , 'password' => bcrypt ('$password'), 'api_token' => '$token']); " | php artisan tinker
"php$php_version" artisan config:clear -n
"php$php_version" artisan config:cache -n
popd
# copy/paste from lychee install, for what it's worth...
# file was touched by artisan, rehash it
# file was touched by artisan, create a new key (copy/paste from Lychee)
ynh_store_file_checksum "$install_dir/.env"
app_key=$(cat $install_dir/.env | grep -e ^APP_KEY | cut -c 9-)
ynh_app_setting_set --key=app_key --value=$app_key
@ -123,18 +66,12 @@ chmod -R 2775 "$install_dir/storage" "$install_dir/app" "$install_dir/public" "$
#=================================================
# SETUP APPLICATION WITH CURL
#=================================================
### Use these lines only if the app installation needs to be finalized through
### web forms. We generally don't want to ask the final user,
### so we're going to use curl to automatically fill the fields and submit the
### forms.
# Installation with curl
ynh_script_progression "Finalizing installation..."
#ynh_local_curl "/INSTALL_PATH" "key1=$name" "key2=$email" "key3=$password" "key4=$password"
# Installation with curl: set up the first/single user
ynh_local_curl "/INSTALL_PATH" "key1=$user" "key2=$email" "key3=$password" "key4=$password"
#=================================================
# END OF SCRIPT
#=================================================
ynh_script_progression "Installation of $app completed"

View file

@ -7,19 +7,6 @@
source _common.sh
source /usr/share/yunohost/helpers
### Settings are automatically loaded as bash variables
### in every app script context, therefore typically these will exist:
### - $domain
### - $path
### - $language
### - $install_dir
### - $port
### ...
### For remove operations:
### - the core will deprovision every resource defined in the manifest **after** this script is ran
### this includes removing the install directory, and data directory (if --purge was used)
#=================================================
# REMOVE SYSTEM CONFIGURATIONS
#=================================================
@ -27,8 +14,6 @@ ynh_script_progression "Removing system configurations related to $app..."
### This should be a symetric version of what happens in the install script
#ynh_config_remove_fail2ban
ynh_config_remove_nginx
#=================================================

View file

@ -15,9 +15,6 @@ ynh_script_progression "Restoring the app main directory..."
ynh_restore "$install_dir"
### $install_dir will automatically be initialized with some decent
### permissions by default... however, you may need to recursively reapply
### ownership to all files such as after the ynh_setup_source step
chown -R "$app:www-data" "$install_dir"
#=================================================
@ -32,15 +29,8 @@ ynh_mysql_db_shell < ./db.sql
#=================================================
ynh_script_progression "Restoring system configurations related to $app..."
### This should be a symetric version of what happens in the install script
ynh_restore "/etc/nginx/conf.d/$domain.d/$app.conf"
#ynh_restore "/etc/fail2ban/jail.d/$app.conf"
#ynh_restore "/etc/fail2ban/filter.d/$app.conf"
#ynh_systemctl --action=restart --service=fail2ban
#=================================================
# RELOAD NGINX AND PHP-FPM OR THE APP SERVICE
#=================================================