/ExampleTicketPHP

Example of a ticket system using PHP

Primary LanguagePHPMIT LicenseMIT

Why we won't want to use a framework?.

A framework is a crucial tool to develop a system quickly and fast. However, it has a cost, the performance. And once the performance is gone, then we must disarm a lot of code to recover the missing performance.

https://miro.medium.com/max/1200/1*QbApXSaxgUvd6b1GM-29-A.png

It is an old comparison, but it is still valid.

Also, a framework is, ahem, a frame, it marks some rules that could work generically, but those rules couldn't be the best for our project.

Reinvent the wheel.

However, we won't want to reinvent the wheel.

In our case, we will use the following libraries:

Also, we want a code that could grow, so it must be easy to maintenance (i.e. we don't want spaghetti code) but also we don't want an over-engineered code.

Since we are working on MVC, then it's easy to determine if a code is clean or not: The Controller. The Controller must do some task but it must not contains the implementation of the task. I explain. Let's say our Controller must read a table, then our controller must call the function that read the table instead of creates the code to read the table inside the controller.

Right example:

function listAction() {
   $r=SomeClassDao::list(); 
}

Bad example:

function listAction() {
   $db=new ConnectionDatabase();
   $r=$db->query('select * from sometable'); 
}

What we want to do?

A ticket system.

What we need to?

Structure

πŸ“œ.htacess
πŸ“œ router.php
πŸ“œ genrepo.php
πŸ“œ composer.json (we use Composer)
πŸ“app
.... πŸ“œ app.php (our common code)
πŸ“controller
.... πŸ“œ TicketController.php
πŸ“ dao
.... πŸ“œ TicketDao.php

πŸ“ repo
.... πŸ“œ AbstractTicketsRepo.php (generated by genrepo.php)

.... πŸ“œ ExampleTicketBase.php (generated by genrepo.php)

.... πŸ“œ TicketsRepo.php (generated by genrepo.php)

πŸ“ factory
.... πŸ“œ TicketFactory.php
πŸ“ views
.... πŸ“ ticket
.... .... πŸ“œ index.blade.php
.... .... πŸ“œ list.blade.php

First, we need some common code and find a way to use it across the whole code. For example, the database. The database must be a singleton (we won't want to create more than one connection per request), and we must configure it once. So, we will use this next function, that works as GLOBAL and singleton generator. It's pretty simple, and it works.

function database() {
    global $database;
    if ($database===null) {
        $database=new PdoOne('mysql','127.0.0.1','root','abc.123','example');
        $database->logLevel=3; // it shows all errors
        try {
            $database->connect(true);
        } catch(exception $ex) {
            die("The database is not available");
        }
    }
    return $database;
}

This function is simple to the extend we could add more code inside it. If the object database doesn't exist, then it creates one. It connects to 127.0.0.1, user root, password abc.123 y schema example.

And here some puritans could claim:

HERESY HERESY!. This code is using global.

And lets me explain something. There is nothing wrong with the use of global. If something must be used or accessed everywhere, then it makes a sense that this something is global. What are we skipping with global? Dependency injection, containers, and whatnot! The use of global is bad if we use GLOBAL for variables or objects that they must not be shared globally, for example, local variables. So the use of global is not as absolute. Only Sith think in absolute 😁 We could use globally if it is used correctly. The 100% ban on the use of GLOBAL it's silly, and it's religion.

Now, I did the same with all the code that must be shared everywhere.

  • valid() for the validation (and error container)
  • database() for the database
  • blade() for the template system
  • router() for the router.

So this solution is simple (so it's easy to debug and to understand), scalable, testable and it's blazing fast!. We could have store those function inside a class but MEH. We don't want to do that. Why?. Because we are adding more code to call the function, nothing more.

If you don't like then you could wrap all of it inside a class, so instead of using this

pdoOne()->method();

you could use (using a class called AppClass and using the database method statically)

AppClass::database()->method();

2-step - Creation of table

For our exercise, we create a new schema called examples.

Inside Mysql, create a schema called examples

Also, we need to create a new table called tickets:

CREATE TABLE `tickets` (
  `IdTicket` INT NOT NULL AUTO_INCREMENT,
  `User` VARCHAR(50) NULL,
  `Title` VARCHAR(50) NULL,
  `Description` VARCHAR(200) NULL,
  PRIMARY KEY (`IdTicket`));

Or using a GUI

3-step Creation of the Repository Layer

The repository layer.

In the root folder, let's create and execute the next code:

πŸ“œ /genrepo.php

It calls a simple method but this method is quite verbose and long.

<?php
include "app/app.php";

$errors=pdoOne()->generateAllClasses(
    ['tickets'=>'TicketsRepo'] // table->class repository
    ,'ExampleTicketBase' // base class (for the database
    ,'eftec\exampleticket\repo' //namespace of the class repository
    ,__DIR__.'/repo'  // folder of the class repository
);

Now, this method creates the next files

  • πŸ“œ /repo/TicketsRepo.php But only if the file does not exist!!. If you want to replace it, then you must delete it manually (or set the argument $forced of the method generateAllClasses() to true). Why? The concept of this class is it could be edited, so if we modify this class, then the changes are keep even if we rebuild all the repositories.
  • πŸ“œ /repo/AbstractTicketsRepo.php It is the class contains all the definitions of the table tickets.
  • πŸ“œ /repo/ExampleTicketBase.php This class is common to all tables

Finally, you can delete πŸ“œ /genrepo.php (optionally)

4-step Router

πŸ“œ /router.php

include "app/app.php";

if (router()->getType()=="controller") {
    try {
        router()->callObject('eftec\exampleticket\controller\%sController', true);
    } catch (Exception $e) {
        echo $e->getTraceAsString();
        echo "<hr>";
        echo "try /Ticket/List to show the table<br>";
        echo "Or /Ticket/Index to insert a new ticket<br>";
    }
}

It is our router file. We use router() (our global function). Our code converts an url into a call to a class/method. For our example:

  • somedomain/Ticket/List -> calls our Ticket Controller Class and the method listAction

  • somedomain/Ticket/Index-> calls our Ticket Controller Class and the method indexActionGet or indexActionPost (if the call is via get or post).

  • somedomain/ -> redirect to somedomain/Ticket/List

It also requires a .htacess file (for Apache)

...
RewriteRule ^(.*)$ router.php?req=$1 [L,QSA]
</IfModule>

Considerations

  • Is our router safe? Yes, if the user enters an incorrect or malicious path, then it will show a message and it only could call a Controller Class, it could not execute any arbitrary code but controller (and methods that end with Action).

5 step - Service class

Now, we need some service class. What is a service-class? A service class in short terms, a collection of methods. Technically it could also contain the (business) logic of the project (business logic = method).

TicketFactory

It's our service class dedicated to creating a new ticket.

  • function factory() In this method we create an empty ticket. It's just an array.
    public static function Factory() {
        return TicketsRepo::factoryNull(); // using the class TicketsRepo created in the step 3.
    }
  • function Fetch() In this method we fetch the values entered by the user, we also validate the information, and we store the errors (if any), inside the validation class. SRP speaking, this method is breaking the SRP. However, fetching->validating->storing messages it's something that works together so separating will not do any good.
public static function Fetch() {
        $ticket=self::Factory(); // we star creating an empty ticket
        $ticket['IdTicket']=Valid()
            ->type('integer')
            ->required(false)
            ->def(0)
            ->condition('gte','The id can\'t be negative',0)
            ->fetch(INPUT_POST,'IdTicket');        
        // etc. with the other fields
        return $ticket;
    }

Our library Valid does the heavy lifting of validating the information of the user. We could have done a simple:

$ticket['IdTicket']=$_POST['IdTicket'];

but what if the information is missing, or if it is not an integer, of it is a string but it is too long, etc. etc.

TicketDao

(obsolete, we will use TicketsRepo generated in the step 3)

It's our service class dedicated to every task involving persistence and tickets.

  • function insert($ticket) This method inserts a new ticket into the database. If the operation fails, then we store a message (inside the validation class).
    public static function insert($ticket) {
        try {
            pdoOne()
                ->set('User=?', $ticket['User'])
                ->set('Title=?', $ticket['Title'])
                ->set('Description=?', $ticket['Description'])
                ->from('Tickets')
                ->insert();
            return true;
        } catch (\Exception $e) {
            valid()->addMessage('ERRORINSERT','Unable to insert. '.pdoOne()->lastError(),'error');
        }
        return false;
    }

We use our global function database(). It uses prepared-statement (hence the Column=? annotation. So it's SQL-injection free. And if it fails, then it will store the message into the valid() method.

  • function list() This method returns all tickets from the database. If the operation fails, then we store a message (inside the validation class).

Considerations

  • What if the database is down?. Then it will stop the execution of our code.
  • What if the operation fails?. Then it will store an error message.
  • What if the user enters a malicious code?. The library uses prepared-statement, so the code is safe from malicious entries. For example: if the name of the ticket is O'hara, the system will work correctly.

So the security and stability of the system are ensured.

6 step - Views

Since we are using the library BladeOne, then we could use views (instead of duct & taping the html+php inside the same class. In our case, we will use two views (for insert and list)

  • πŸ“œ /views/ticket/index It is part of the view: (without using eftec/BladeOneHtml)
<input type="text" name="User" class="form-control mb-4" placeholder="User" value="{{$ticket['User']}}">
@if(valid()->getMessageId('User')->countError())
	<div class="text-danger">\{\{ valid()->getMessageId('User')->first()\}\}<br></div>
@endif()

And using the extension eftec/BladeOneHtml

@input(type="text" name="User" class="form-control mb-4" placeholder="User" value=$ticket['User'])
@if(valid()->getMessageId('User')->countError())
    <div class="text-danger">{{valid()->getMessageId('User')->first()}}<br></div>
@endif()

Here we are doing two operations: we are showing the $ticket (field User), and we are showing an error message (if any) using our global function called valid(). Each error is store into a container (in this case the container is called "User"). So it is possible to show more than one error at the same time (and this project does that).

  • πŸ“œ /views/ticket/list It shows a list of tickets.

Considerations

  • Is the view safe?. Yes, It uses { { $variable} } to show a variable. It is converted into htmlentities($variable)

7 step Controller

What is a controller class?. In short, it's a class called by the router and it joins all the code. Finally, the controller class could call the view layer (and in our case, it's exactly what it does).

<?php
namespace eftec\exampleticket\controller;


use eftec\exampleticket\factory\TicketFactory;

class TicketController
{
    public static function HomeAction($id="",$idparent="",$event="") {
    }
    public static function IndexActionGet($id="",$idparent="",$event="") {
    }
    public static function IndexActionPost($id="",$idparent="",$event="") {
    }
    public static function ListAction($id="",$idparent="",$event="") {
    }
}

It is our controller (without implementation)

  • HomeAction (is called for GET and POST) as a default action
  • IndexActionGet (is called for GET) when we want to insert a new ticket
    public static function IndexActionGet($id="",$idparent="",$event="") {
        $ticket=TicketFactory::Factory(); // we create a new ticket.
        echo blade()->run('ticket.index',['ticket'=>$ticket]);
    }

This method is called when we want to show a new form to insert a new TICKET. So we create a new ticket (an empty ticket). How? We use the TicketFactory::Factory() method. Finally, we called the view ticket/index, and we send the ticket (we want to show a TICKET even if it is empty). In our view, it looks like :

value="$ticket['User']">

Note: what is the meaning of the arguments: ($id="",$idparent="",$event="") ? Those arguments could be use by our code. They are filled by the router(). In this case, if we call the route : domain/Ticket/Index/1/2?_event=hi then $id=1, $idparent=2 and $event=hi.

  • IndexActionPOST (is called for POST) when we want to insert a new ticket and after we push the button
    public static function IndexActionPost($id = "", $idparent = "", $event = "") {
        $ticket = TicketFactory::Fetch(); // we read the information obtained from the user
        if (blade()->csrfIsValid()) {
            if (valid()->messageList->errorcount === 0) {
                if (TicketsRepo::insert($ticket)) {
                    // ticket inserted correctly, let's go to the list
                    header('Location: ../../Ticket/List');
                    exit();
                }
                echo blade()->run('ticket.list', ['tickets' => $ticket]);
            } else {
                echo blade()->run('ticket.index', ['ticket' => $ticket]);
            }              
        } else {
            echo "csrf invalid";
        }
    }

We call this method after we push the button (POST).

  • First, we read (fetch) the information entered by the user. If the information has errors, then the error messages are stored inside valid()

  • Second, we check valid. If the error count is zero then we insert the ticket inside the database. And if the we insert it correctly then redirect to the list.

  • Third, however if something fails, then we show the form again (and the view shows any error).

  • ListAction (is called for GET and POST) when we want to show the list

8 step - And finally, it is our code

http://localhost/ExampleTicketPHP/Ticket/List

Or change the URL according your server and folders.

The full code is here

https://github.com/jorgecc/ExampleTicketPHP

How much minimalist is our code?

  • 34 PHP files (including libraries and examples).
  • 7000 lines of code (including lines of code of our libraries).
  • Our code (excluding libraries and views) are 166 lines of code. :-3

What is missing?

  • Since this project lacks users, then it misses cross-posting protection.

So, let's add csrf protection.

It is our controller with csrf protection:

    public static function IndexActionGet($id="",$idparent="",$event="") {
        // ....
        blade()->regenerateToken(); // we create a new csrf token
    }
    public static function IndexActionPost($id="",$idparent="",$event="") {
        // .....
        if (blade()->csrfIsValid()) { // we validated the token
           // ....
        } else {
            valid()->addMessage('TOKEN','Token invalid','error');
        }
        // ....
    }

and in our view

<form class="border border-light p-5" method="post">
    @csrf()
   ....
  • It also misses cache and pagination. We won't add cache or pagination but we want to limit the number of tickets to show.

It is part of the code without limit

$tickets = pdoOne()->select('*')
	->from('Tickets')
	->toList(); // select * from tickets

It is part of the code with limit, so the system will never overflow.

$tickets = pdoOne()->select('*')
	->from('Tickets')
	->order('IdTicket desc')
	->limit('1,20')
	->toList(); // select * from tickets order by idticket desc limit 1,20