rougin / basilisk
A project skeleton for Slytherin.
Requires
- php: >=5.4.0
- illuminate/database: ~5.0|~6.0|~7.0|~8.0|~9.0|~10.0|~11.0
- illuminate/events: ~5.0|~6.0|~7.0|~8.0|~9.0|~10.0|~11.0
- illuminate/view: ~5.0|~6.0|~7.0|~8.0|~9.0|~10.0|~11.0
- robmorgan/phinx: ~0.8
- rougin/weasley: ~0.6
- vlucas/phpdotenv: ~2.0
Requires (Dev)
- phpunit/phpunit: ~4.2|~5.7|~6.0|~7.0|~8.0|~9.0
- sanmai/phpunit-legacy-adapter: ~6.1|~8.0
This package is auto-updated.
Last update: 2024-09-09 07:27:51 UTC
README
Basilisk is a skeleton project made for Slytherin which provides a code structure that is based on my experiences creating projects using Slytherin as it's framework. The said code structure should be easy to understand and be based on SOLID principles.
Installation
Create a new project using Basilisk
via Composer:
$ composer create-project rougin/basilisk "hogwarts"
What's inside?
Directory Structure
Note
The following directory names below are only the preferred names based on my experience building projects under Slytherin. But they can be easily be extended or removed as Slytherin not does not conform to those said preferences.
src/
├─ Checks/
├─ Depots/
├─ Models/
├─ Phinx/
│ ├─ Scripts/
│ ├─ Seeders/
├─ Routes/
├─ Scripts/
Checks
src/
├─ Checks/
This directory contains classes that are used for validation. Those classes may be extended to the Check
class of Weasley:
namespace App\Checks; use Rougin\Weasley\Check; class UserCheck extends Check { /** * @var array<string, string> */ protected $labels = [ 'name' => 'Name', 'email' => 'Email', ]; /** * @var array<string, string> */ protected $rules = [ 'name' => 'required', 'email' => 'required|email', ]; }
Depots
src/
├─ Depots/
The main directory that should contain the logic of a project:
namespace App\Depots; use App\Models\User; class UserDepot { /** * @var \App\Models\User */ protected $user; /** * @param \App\Models\User $user */ public function __construct(User $user) { $this->user = $user; } /** * @return array<string, mixed>[] */ public function all() { $result = $this->user->all(); $items = array(); foreach ($result as $item) { $row = ['id' => $item->id]; $row['name'] = $item->name; $row['email'] = $item->email; $items[] = $row; } return (array) $items; } }
Note
Prior in using depots, or may be best known as the Repository pattern, I implemented most of the logic in the Routes
or Models
directories. However, it presents a challenge to me in organizing their code. With using depots, I can reuse the same logic in to either Routes
(for receiving user request) or in Scripts
directory (for handling terminal-based actions).
Models
src/
├─ Models/
It is the directory where for storing the models (for Eloquent
) or entities (if using Doctrine
). The classes in this directory should represent a database table (e.g., if having a users
table, it should be represented as User
class):
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; class User extends Model { /** * @var string[] */ protected $fillable = [ 'name', 'password', 'email', ]; /** * @var string[] */ protected $hidden = [ 'password' ]; /** * @var string */ protected $table = 'users'; }
Phinx
src/
├─ Phinx/
│ ├─ Scripts/
│ ├─ Seeders/
This directory is for the storage of related files for the Phinx
package. The Scripts
directory contains the generated database migrations while the Seeders
directory must contain the database seeders.
// src/Phinx/Scripts/20171012020230_create_users_table.php use Phinx\Migration\AbstractMigration; class CreateUsersTable extends AbstractMigration { public function change(): void { $properties = ['id' => false, 'primary_key' => ['id']]; $table = $this->table('users', $properties); $table->addColumn('id', 'integer', ['limit' => 10, 'identity' => true]); $table->addColumn('name', 'string', ['limit' => 200]); $table->addColumn('email', 'string', ['limit' => 200]); $table->addColumn('password', 'string', ['limit' => 500]); $table->addColumn('created_at', 'datetime'); $table->addColumn('updated_at', 'datetime', ['null' => true]); $table->create(); } }
// src/Phinx/Seeders/UserSeeder.php use Phinx\Seed\AbstractSeed; class UserSeeder extends AbstractSeed { protected $items = [ ['name' => 'Harry Jonathans Potter', 'email' => 'hjpotter@hogwarts.co.uk'], ['name' => 'Hermione Jane Granger', 'email' => 'hjgranger@hogwarts.co.uk'], ['name' => 'Ronald Bilius Weasley', 'email' => 'rbweasley@hogwarts.co.uk'], ]; public function run(): void { $data = array(); foreach ($this->items as $item) { $item['created_at'] = date('Y-m-d H:i:s'); $data[] = $item; } $this->table('users')->insert($data)->save(); } }
The said directory will be used for running the database migrations and seeders. To migrate the database scripts, kindly run the migrate
command:
$ vendor/bin/phinx migrate -c app/config/phinx.php
Note
Before running scripts from Phinx
, kindly update the database credentials first in .env
:
$ cp .env.example .env
The seed:run
command can be used for populating data in a database:
$ vendor/bin/phinx seed:run -c app/config/phinx.php
Note
The command above will load the seeders in alphabetical order.
Routes
src/
├─ Routes/
The gateway of the project wherein the routes are stored. The said class can call or instantiate the classes found from the previously mentioned directories.
namespace App\Routes; use Rougin\Slytherin\Template\RendererInterface; class Hello { /** * Returns the "Hello, Muggle!" text. * * @return string */ public function index(RendererInterface $renderer) { return $renderer->render('index'); } }
Note
In other frameworks, Routes
is commonly known as Controllers
.
Scripts
src/
├─ Scripts/
The directory where the scripts are stored. These scripts can be executed directly using the php
command in the terminal:
// src/Scripts/HelloWorld.php echo 'Hello world!';
$ php src/Scripts/HelloWorld.php
Hello world!
Running the application
The PHP's built-in web server can be used for running the project during development:
$ php -S localhost:8000 -t app/public
Then open a web browser and proceed to go to http://localhost:8000.
Changelog
Please see CHANGELOG for more information what has changed recently.
Testing
$ vendor/bin/phpunit
License
The MIT License (MIT). Please see LICENSE for more information.