Skip to content

This is a reference project elaborated by the students step-by-step in every FHNW web engineering lecture.

License

Notifications You must be signed in to change notification settings

webengfhnw/WE-CRM

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

WE-CRM

This is a reference project elaborated by the students step-by-step in every FHNW web engineering lecture.

Analysis

Scenario

WE-CRM (Web Engineering Customer-Relationship-Management) is the smallest possible and lightweight demonstration tool that allows agents to manage their customer data. Agents have an own access to their customer data. Besides, agents can email themselves an complete extract or create a PDF of their customers.

Use Case

  • UC-1 [Login on WE-CRM]: Agents can log-in by entering an email address and password. As an extension, new agents my register first.
  • UC-2 [Register on WE-CRM]: Agents can register to get an account (profile) to access the WE-CRM system.
  • UC-3 [Edit a customer]: Agents can create, update and delete customers.
  • UC-4 [Show a customer list]: Agents can get an overview over their customers based on a customer list. As an extension they can create, update and delete customers (UC-3), generate a PDF (UC-5) or send an email (UC-6).
  • UC-5 [Generate a PDF customer list]: Agents can generate a PDF containing a list of their customers.
  • UC-6 [Send customer list via email]: Agents can send an email containing a list of their customers to their own inbox.

Design

Information Systems Modelling

Information Systems (Layering) Architecture

Implementation

Stage 1: Building a Static Website with Bootstrap

In stage 01 a bootstrap based prototype has been created by using a prototyping application.

Wireframes

Log-in Customers Edit

HTML-Prototype

In this case, the prototype application Bootstrap Studio has been used to create a basic user interface design based on an HTML grid, Bootstrap CSS and JavaScript, including the selection of web fonts and font-based icons.

The assets (HTML, CSS, JavaScript, image and font files) has been exported and will be extended in the later stages by PHP logic, and later with jQuery, to build a dynamic website.

Stage 2: PHP Files and Session

In stage 02 the HTML prototype files will be transferred to PHP files, and a basic session functionality will be implemented.

Session

Sessions are an almost secure (not 100%) way to identify a user over several requests.

"To help mitigate cross-site scripting (XSS) attacks, HttpOnly cookies are inaccessible to JavaScript's Document.cookie API; they are only sent to the server." see: MDN web docs - HTTP cookies

Therefore, it is recommended to start a session at the beginning of a PHP script including the definition to have httponly session cookies only as follows:

ini_set( 'session.cookie_httponly', 1 );
session_start();

Then a session value (such as a user id) can be stored in the session array:

$_SESSION["key"] = "value";

It is recommended to regenerate a session id after every authentication state change to prevent session fixation:

session_regenerate_id(true);

And a value can be accessed again:

$value = $_SESSION["key"];

Finally, a session can be destroyed again if required (such as logout):

session_destroy();

Stage 3: namespace/use, Auto-Loading, .htaccess and Router

In stage 3, namespaces and the usage statement are substituting the explicit definition of require and include.

namespace/use and Auto-Loading

In PHP a namespace-use scenario is not 100 % equivalent to the package-import scenario is for instance in Java. Namespaces in PHP are providing a similar mechanism for structuring source code, which is stored in separate files, using a namespace, package or folder-like structure. With the use statement, code within a namespace can be accessed if, and this is not equivalent to Java, for instance, a separated code part is included (require and include).

It is advisable that an autoloader is implemented or a project is relying on a composer-related mechanism such as PSR-4, to overcome the separation of namespaces and files. The following autoloader links the directory structure and a namespace.

namespace config;
class Autoloader
{
    public static function autoload($className) {
        //replace namespace backslash to directory separator of the current operating system
        $className = str_replace('\\', DIRECTORY_SEPARATOR, $className);
        $fileName = $className . '.php';

        if (file_exists($fileName)) {
            include_once($fileName);
        }
    }
}

This own-written autoloader can then be registered using a PHP built-in function:

spl_autoload_register('config\Autoloader::autoload');

.htaccess

The following .htaccess configuration ensures that HTTPS is used (except on localhost) and redirects everything (except asset requests) to the index.php file:

# .htaccess files provide a way to make configuration changes on a per-directory basis
RewriteEngine On

# this ensures that HTTPS is used except on localhost
RewriteCond %{HTTP_HOST} !^localhost(?::\d+)?$ [NC]
RewriteCond %{HTTPS} off
RewriteCond %{HTTP:X-Forwarded-Proto} !https
RewriteRule ^ https://%{HTTP_HOST}%{REQUEST_URI} [QSA,L,R=301]

# this sends the authorization header to a PHP envirnoment variable
RewriteRule .* - [E=HTTP_AUTHORIZATION:%{HTTP:Authorization}]

# this redirects everything except asset requests to the index.php file
RewriteRule ^(?!.*assets/)(.*) index.php [QSA,L,E=ORIGINAL_PATH:/$1]
RewriteRule assets/(.*) view/assets/$1 [QSA,L]

Router

The router provides redirection, an error header, the PATH_INFO and a ROOT_URL global. Then, the link structure has been adapted according to the routers (router configuration) using the ROOT_URL global if required.

The following route_auth function stores a route (the configured path) in a multidimensional array using the HTTP method and the path. The route consists of an authentication and a route callback function.

public static function route_auth($method, $path, $authFunction, $routeFunction) {
    if(empty(self::$routes))
        self::init();
    $path = trim($path, '/');
    self::$routes[$method][$path] = array("authFunction" => $authFunction, "routeFunction" => $routeFunction);
}

The following call_route function is used to process every request.

public static function call_route($method, $path, $errorFunction) {
    $path = trim(parse_url($path, PHP_URL_PATH), '/');
    if(!array_key_exists($method, self::$routes) || !array_key_exists($path, self::$routes[$method])) {
        $errorFunction(); return;
    }
    $route = self::$routes[$method][$path];
    if(isset($route["authFunction"])) {
        if (!$route["authFunction"]()) {
            return;
        }
    }
    $route["routeFunction"]();
}

The router provides the possibility to register routes using a static method:

 Router::route("POST", "/login", function () {
    /** TODO */
    Router::redirect("/");
});

Besides, it is possible to set an authentication function before the callback function is executed:

Router::route_auth("GET", "/", $authFunction, function () {
    /** TODO */
    layoutSetContent("customers.php");
});

Finally, the router will be called by defining the following method in the entry point of the web application, which is, in this case, the index.php file:

Router::call_route($_SERVER['REQUEST_METHOD'], $_SERVER['PATH_INFO'], $errorFunction);

Stage 4: Database, .env Config Files and Passwords

In stage 4 (and stage 5) WE-CRM will be extended with a database functionality.

Entity Relationship Diagram

Database

As a first step, an Entity Relationship Diagram needs to be created, which can be partially be derived from the use case's nouns.

Depending on the modelling environment, a Domain Model can be created in-sync at the same time. Please make sure that "in" parameter direction is configured. The Domain Model will be used in stage 7 to implement a basic object-relational data access using PDO.

As a result and depending on the modelling environment (Visual Paradigm Postgresql Database Generation), a Data Definition Language (DDL) SQL can be exported as follows:

CREATE TABLE Customer (
  ID      SERIAL NOT NULL, 
  Name    varchar(255) NOT NULL, 
  Email   varchar(255) NOT NULL, 
  Mobile  varchar(255) NOT NULL, 
  AgentID int4 NOT NULL, 
  PRIMARY KEY (ID));
CREATE TABLE Agent (
  ID       SERIAL NOT NULL, 
  Name     varchar(255) NOT NULL, 
  Email    varchar(255) NOT NULL, 
  Password varchar(255) NOT NULL, 
  PRIMARY KEY (ID));
CREATE TABLE AuthToken (
  ID         SERIAL NOT NULL, 
  AgentID    int4 NOT NULL, 
  Selector   varchar(255) NOT NULL, 
  Validator  varchar(255) NOT NULL, 
  Expiration timestamp NOT NULL, 
  Type       int4 NOT NULL, 
  PRIMARY KEY (ID));
ALTER TABLE Customer ADD CONSTRAINT AgentCustomer FOREIGN KEY (AgentID) REFERENCES Agent (ID);
ALTER TABLE AuthToken ADD CONSTRAINT AgentToken FOREIGN KEY (AgentID) REFERENCES Agent (ID);

As a result of this stage, the user (agent) registration and login will be realized using (Sessions), (.env Config Files), (PDO) and dealing with (Passwords) securely.

.env Config Files

As a best practice, database related configuration should be stored outside of the source code in a configuration file. By convention, .env files must be kept outside of a version control by adding an entry to .gitignore. In this web application, the database configuration will be loaded from an INI file with an .env extension, since PHP provides already integrated functions for reading INI files. The file in the config folder may look like this:

[database]
database.dsn="pgsql:host=<host>;port=<port>;dbname=<database>;sslmode=require"
database.user=<user>
database.password=<password>

To read such an INI file, the following PHP functions can be used:

class Config
{
    protected static $iniFile = "config/config.env";
    protected static $config = [];

    public static function init()
    {
        if (file_exists(self::$iniFile)) {
            self::$config = parse_ini_file(self::$iniFile);
        } else if (file_exists("../". self::$iniFile)) {
            self::$config = parse_ini_file("../". self::$iniFile);
        } else {
            self::loadENV();
        }
    }
    // ...
}

If this application is deployed out of this GitHub repository to Heroku, the Heroku app can be extended with a Postgresql database. The configuration items to this database can be accessed from PHP code, which is running on Heroku, by using environment variables as follows:

private static function loadENV(){
    if (isset($_ENV["DATABASE_URL"])) {
        $dbopts = parse_url($_ENV["DATABASE_URL"]);
        self::$config["database.dsn"] = "pgsql" . ":host=" . $dbopts["host"] . ";port=" . $dbopts["port"] . "; dbname=" . ltrim($dbopts["path"], '/') . "; sslmode=require";
        self::$config["database.user"] = $dbopts["user"];
        self::$config["database.password"] = $dbopts["pass"];
    }
}

The config class consists of a static method delivering configuration parameters defined in the .env file (INI-based syntax):

Config::get("database.dsn");

PDO

As a next step, the user (agent) registration and login are realized using PDO for data access.

In order to use PDO with Postgresql the following lines need to be un-commented in php.ini:

extension=php_pdo_pgsql.dll
extension=php_pgsql.dll

As a good practice, the PDO instantiation should be kept in a different class containing static methods.

The database class has been implemented to hold an instance of PDO. This instance will be created once if the class attribute is NULL - this can be realized as following:

protected function __construct()
{
    self::$pdoInstance = new PDO (Config::get("database.dsn"), Config::get("database.user"), Config::get("database.password"));
    self::$pdoInstance->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
}

Then prepared statements can be executed. In the following example an associative array (PDO::FETCH_ASSOC) will be returned. In the later stage 7, objects will be mapped to tables (PDO::FETCH_CLASS):

$stmt = $pdoInstance->prepare('SELECT * FROM table WHERE id = :id;');
$stmt->bindValue(':id', $id);
$stmt->execute();
$resultArray = $stmt->fetchAll(PDO::FETCH_ASSOC);

Passwords

Passwords are extremely sensitive data.

  1. They must be transmitted over HTTPS only - never HTTP only!
  2. Always the best hashing method available in PHP must be used before storing a password in a database table.
  3. Try to keep the raw / un-hashed password as short as possible in memory.

A secure password hashing in PHP can be realized as follows:

$hashedPassword = password_hash($_POST["password"], PASSWORD_DEFAULT);
// store the $hashedPassword in DB

A secure password verification with a re-hashing if required can be realized as follows:

if (password_verify($_POST["password"], $hashedPassword)) {
    // start session
    if (password_needs_rehash($hashedPassword, PASSWORD_DEFAULT)) {
        $reHashedPassword = password_hash($_POST["password"], PASSWORD_DEFAULT);
        // store the $reHashedPassword in DB
    }
}

Stage 5: Dynamic Views

In stage 5, the web-application is extended with functionality to store and retrieve customer data similar as described in the (PDO) section.

The view files are extended with <?php ?> tags injecting the required dynamic data. The following example shows how a HTML table can be dynamically populated:

<?php foreach($customers as $customer): ?>
    <tr>
        <td><?php echo $customer["id"] ?> </td>
        <td><?php echo $customer["name"] ?></td>
        <td><?php echo $customer["email"] ?> </td>
        <td><?php echo $customer["mobile"] ?> </td>
    </tr>
<?php endforeach; ?>

The following code snipped shows how an HTML form input field value can be set, if data is available:

<input class="form-control" type="email" name="email" value="<?php echo !empty($customer["email"]) ? $customer["email"] : ''; ?>">

Stage 6: Exception Handling and HTTP Status

Stage 6 contains classes for exception handling. Exceptions are an adequate way to handle with errors. The exceptions are thrown using a throw statement followed by an exception class:

function inverse($x) {
    if (!$x)
        throw new Exception('Division by zero.');
    return 1/$x;
}

Using the try-catch structure, an exception can be handled using the thrown object:

try {
    inverse(0);
} catch (Exception $e) {
    echo 'Caught exception: ',  $e->getMessage();
}

Since PHP is an HTTP-oriented programming environment, it makes sense to use the HTTP status codes for dealing with exceptions. The file HTTPStatusHeader.php contains a list of constants representing common HTTP status codes (based on Henrique Moody derived from Wikipedia - List of HTTP status codes). Since HTTP status codes are used in standard header definition such as redirects, an HTTPHeader class has been implemented as well. Therefore the HTTPException and the HTTPHeader implementations are using the same HTTP status codes methods, which requires the use of Traits. Traits are an elegant way to overcome typical problems associated with multiple inheritances. The following incomplete and combined listing show the power of Traits:

interface HTTPStatusCode
{
   const HTTP_200_OK = "OK";
   // ...
}

trait HTTPStatusHeader
{
   public static function setStatusHeader($statusCode = HTTPStatusCode::HTTP_200_OK, $replaceHeader = true, $statusPhrase = null){
       // ...
   }

   public static function setHeader($header, $statusCode = HTTPStatusCode::HTTP_200_OK, $replaceHeader = true){
       // ...
   }

}

class HTTPHeader implements HTTPStatusCode
{
   use HTTPStatusHeader;

   public static function redirect($redirect_path, $statusCode = HTTPStatusCode::HTTP_301_MOVED_PERMANENTLY) {
       // ...
   }
}

class HTTPException extends Exception implements HTTPStatusCode
{
   use HTTPStatusHeader;

   public function __construct($statusCode = HTTPStatusCode::HTTP_400_BAD_REQUEST, $statusPhrase = null, $body = null)
   {
       self::createHeader($statusCode, $statusPhrase);
       // ...
   }

   public function getHeader($replaceHeader = true){
       self::setHeader($this->header, $this->statusCode, $replaceHeader);
       return $this->header;
   }
}

As shown above, the HTTPException class has been extended from Exception and trough the Trait functionality with the HTTPStatusHeader as well.

As a result of the new HTTPException class, the router is adapted to throw exceptions, and the router configuration now deals with the thrown exception by showing the corresponding HTTP status code:

try {
    Router::call_route($_SERVER['REQUEST_METHOD'], $_SERVER['PATH_INFO']);
} catch (HTTPException $exception) {
    $exception->getHeader();
}

Stage 7: Domain and Data Access Objects (DAO)

In stage 7, the Domain Model, which has been elaborated in stage 4 (database section), will be transferred into PHP code and be accessible by implementing data access objects (DAO) in a CRUD (create read, update and delete) style.

Domain Model

Domain Objects

Depending on the modelling environment, a Domain Model can be transferred into PHP classes. Such domain objects contain no logic, except some very basic logic for setting and getting data (getters/setters).

PDO can provide a very basic object-relational data access mechanism. In the following example the PDO::FETCH_CLASS strategy is used to provide an array of domain\Agent, which will be used in the next DAO section:

$stmt = $this->pdoInstance->prepare('
    SELECT * FROM agent WHERE email = :email;');
$stmt->bindValue(':email', $email);
$stmt->execute();
$agent = $stmt->fetchAll(PDO::FETCH_CLASS, "domain\Agent")[0];

Data Access Model

Data Access Objects (DAO)

Data Access Objects (DAOs) are classes, which consist of CRUD-like (create, read, update & delete) methods as shown in the Data Access Model. Beside the CRUD methods, DAO usually consists of find methods as well. Depending on the selected strategy, DAOs are related to Domain Objects, since almost all domain object has a corresponding DAO.

It is more popular nowadays to use the term repository for DAOs. But these two concepts are not the same. DAOs are used to build an abstraction layer for data access only. Repositories are implemented on a higher abstraction for a collection of objects. Many frameworks are implementing a repository using the DAO pattern, whereas vice-versa is not possible. Although this is debatable too, since using the definition of by Edward Hieatt and Rob Mee repositories should exist as an mediating object: "[A repository mediates] between the domain and data mapping layers using a collection-like interface for accessing domain objects."

There is a big debate if DAOs should consist of static methods or not. Although DAOs are not designed on the same abstraction level such as repositories, DAOs should be designed to reflect various database connections and drivers generically. Therefore DAOs should be implemented as objects to inject different database connections and drivers at run-time.

The Data Access Model is the blueprint of the DAOs in this reference project. After generating the DAO classes, they will be extended with prepared database statements using PDO as shown here:

namespace dao;

use domain\Agent;

class AgentDAO extends BasicDAO {

	public function findByEmail($email) {
        $stmt = $this->pdoInstance->prepare('
            SELECT * FROM agent WHERE email = :email;');
        $stmt->bindValue(':email', $email);
        $stmt->execute();
        if ($stmt->rowCount() > 0)
            return $stmt->fetchAll(\PDO::FETCH_CLASS, "domain\Agent")[0];
        return null;
    }
}

Stage 8: Business Services

In stage 8, two service interfaces and service implementations based on the Business Logic Model are given. In this reference project, the term service referred to business service has been selected. Sometimes the term business logic acting on a business logic layer is used.

Business Logic Model

Service Interfaces and Implementations

Since this reference project is as small and simplified as possible for teaching purposes, a one-class singleton strategy has been chosen for authentication (AuthService) and one service has been implemented for the customer use-cases (CustomerService). In a bigger application scenario, it would make sense to build several business services for different use-cases.

Business services contain the business decisions as in PHP transferred business rules. An example such a rule could be that an agent is only allowed to edit the customers she or he is responsible for.

As shown in the Business Logic Model, the service interface just lists the required methods, which must be implemented in the service implementation.

It is rarely the case that the usage of a singleton makes sense in PHP, since, compared to Java, PHP is strictly stateless (except session) and every request is in an isolated process. Nevertheless, in this reference project the authentication service implementation is realized using the singleton pattern:

class AuthServiceImpl implements AuthService {

    private static $instance = null;

    private $currentAgentId;

    public static function getInstance() {
        if (!isset(self::$instance)) {
            self::$instance = new self();
        }
        return self::$instance;
    }

    protected function __construct() { }

    private function __clone() { }

    protected function verifyAuth() {
        if(isset($this->currentAgentId))
            return true;
        return false;
    }
}

The business logic, which is referred to the customer use-case(s), is implemented in the customer service:

class CustomerServiceImpl implements CustomerService
{
    public function createCustomer(Customer $customer) {
        if(AuthServiceImpl::getInstance()->verifyAuth()) {
            $customerDAO = new CustomerDAO();
            $customer->setAgentId(AuthServiceImpl::getInstance()->getCurrentAgentId());
            return $customerDAO->create($customer);
        }
        throw new HTTPException(HTTPStatusCode::HTTP_401_UNAUTHORIZED);
    }
}

Stage 9: Template View Pattern and XSS

Since this reference project does not rely on a template engine such as Blade or Twig, a template view pattern class will be implemented in stage 9.

Template View Pattern

The one template view pattern does not really exist. This implementation has been inspired by the Book PHP Design Patterns and the example of Alejandro Gervasio.

In following the TemplateView class is explained - the complete TemplateView class can be found within the stage09\view folder.

The basic idea is to assign a view .php file to a view by passing the information through the constructor:

$contentView = new TemplateView("customerEdit.php");

class TemplateView {

    private $view;

    public function __construct($view) {
        $this->view = $view;
    }
}

Once the view has been instantiated, data can be injected into the view by using a magic __set() method. Finally, the view will be rendered by using the render() method:

$contentView = new TemplateView("customerEdit.php");
$contentView->customer = (new CustomerServiceImpl())->readCustomer($id);
echo $contentView->render();

class TemplateView {

    private $view;
    private $variables = array();

    public function __set($key, $variable) {
        $this->variables[$key] = $variable;
    }

    public function render() {
        extract($this->variables);
        ob_start();
        require($this->view);
        return ob_get_clean();
    }
}

The data that has been injected can be accessed within a view .php file by using a magic __get() method. At the same time it may make sense to validate if a variable has been set:

isset($this->customer) ? $customer = $this->customer : $customer = new Customer();
<input class="form-control" type="text" name="id" readonly="" value="<?php echo $customer->getId() ?>">

class TemplateView {

    private $variables = array();

    public function __get($key) {
        return $this->variables[$key];
    }

    public function __isset($key) {
        if(!array_key_exists($key, $this->variables))
            return false;
        return isset($this->variables[$key]);
    }
}

XSS

To prevent XSS (Cross-Site Scripting) attacks any character in a user input that can affect the structure of the HTML document must be escaped on output (when displaying to a user). Following the guidelines of the Paragon Initiative Enterprises Blog the TemplateView class consists of a static method that can be used in a view .php file:

<input class="form-control" type="text" name="name" value="<?php echo TemplateView::noHTML($customer->getName()) ?>">

class TemplateView {

    public static function noHTML($input, $bEncodeAll = true, $encoding = "UTF-8")
    {
        if($bEncodeAll)
            return htmlentities($input, ENT_QUOTES | ENT_HTML5, $encoding);
        return htmlspecialchars($input, ENT_QUOTES | ENT_HTML5, $encoding);
    }
}

Stage 10: Model-View-Controller

Model–view–controller (MVC) is a software architectural pattern, which is one of the most quoted pattern and at the same time it is the most misquoted pattern. There exist various variants of MVC and variants, which are related to MVC such as model-view-presenter (MVP).

  • The model stores the data, which will be displayed in the view. In general, the model expresses the behaviour by data, logic and rules.
  • The view generates the output to the user based on the model data.
  • A controller triggers actions such as a model update, loads an associated view and processes a certain user interaction.

This reference project reflects the following recommendations of Martin Fowler:

  • Make a strong separation between presentation (view & controller) and domain (model) - separated presentation.
  • Divide user interface into a controller (for reacting to user stimulus) and view (for displaying the state of the model). Controller and view should (mostly) not communicate directly but through the model.

In the following the in stage 10 implemented CustomerController:

class CustomerController
{
    public static function create(){
        $contentView = new TemplateView("customerEdit.php");
        LayoutRendering::basicLayout($contentView);
    }

    public static function readAll(){
        $contentView = new TemplateView("customers.php");
        $contentView->customers = (new CustomerServiceImpl())->findAllCustomer();
        LayoutRendering::basicLayout($contentView);
    }

    public static function edit(){
        $id = $_GET["id"];
        $contentView = new TemplateView("customerEdit.php");
        $contentView->customer = (new CustomerServiceImpl())->readCustomer($id);
        LayoutRendering::basicLayout($contentView);
    }

    public static function update(){
        $customer = new Customer();
        $customer->setId($_POST["id"]);
        $customer->setName($_POST["name"]);
        $customer->setEmail($_POST["email"]);
        $customer->setMobile($_POST["mobile"]);
        if ($customer->getId() === "") {
            (new CustomerServiceImpl())->createCustomer($customer);
        } else {
            (new CustomerServiceImpl())->updateCustomer($customer);
        }
    }

    public static function delete(){
        $id = $_GET["id"];
        (new CustomerServiceImpl())->deleteCustomer($id);
    }
}

The PHP code above uses a static method of the LayoutRendering class, which provides the basic layout of the reference project:

namespace view;

class LayoutRendering
{
    public static function basicLayout(View $contentView){
        $view = new TemplateView("layout.php");
        $view->header = (new TemplateView("header.php"))->render();
        $view->content = $contentView->render();
        $view->footer = (new TemplateView("footer.php"))->render();
        echo $view->render();
    }
}

Stage 11: Validation

In stage 11, a PHP input field validator is implemented. Validation refers to the possibility to verify certain fields such as an email field containing a valid email address ([email protected]). Validation can be realised on the client and back-end side. This PHP validation is realised on the PHP back-end, by implementing domain-specific validation classes:

class CustomerValidator
{
    private $valid = true;
    private $emailError = null;

    public function __construct(Customer $customer = null)
    {
        if (!is_null($customer)) {
            $this->validate($customer);
        }
    }

    public function validate(Customer $customer)
    {
        if (!is_null($customer)) {
            if (!filter_var($customer->getEmail(), FILTER_VALIDATE_EMAIL)) {
                $this->emailError = 'Please enter a valid email address';
                $this->valid = false;
            }
        } else {
            $this->valid = false;
        }
        return $this->valid;

    }

    public function isValid()
    {
        return $this->valid;
    }

    public function isEmailError()
    {
        return isset($this->emailError);
    }

    public function getEmailError()
    {
        return $this->emailError;
    }
}

Such a validator can then be used in a controller to verify if the provided date is valid:

class CustomerController
{
    public static function update(){
        $customer = new Customer();
        // ...
        $customerValidator = new CustomerValidator($customer);
        if($customerValidator->isValid()) {
            // ...
        }
        else{
            $contentView = new TemplateView("customerEdit.php");
            $contentView->customer = $customer;
            $contentView->customerValidator = $customerValidator;
            LayoutRendering::basicLayout($contentView);
            return false;
        }
        return true;
    }

}

If data is invalid, error messages can be displayed:

<div class="form-group <?php echo $customerValidator->isEmailError() ? "has-error" : ""; ?>">
    <div class="input-group">
        <div class="input-group-addon"><span>Email </span></div>
        <input class="form-control" type="email" name="email" value="<?php echo TemplateView::noHTML($customer->getEmail()) ?>">
    </div>
    <p class="help-block"><?php echo $customerValidator->getEmailError() ?></p>
</div>

Stage 12: Auth and Remember Me

In this stage 12, the token-based securing of the business services will be extended, and a remember me functionality will be implemented. The business services are secured using an internal token, which is bound to the session, since stage 09. In this stage, the generated token will be securely stored in the database. Therefore an AuthToken domain object (as shown in the Domain Model) with a corresponding AuthTokenDAO (as shown in the Data Access Model) will be crated. The AuthToken database table has been implemented as follows:

CREATE TABLE AuthToken (
  ID         SERIAL NOT NULL, 
  AgentID    int4 NOT NULL, 
  Selector   varchar(255) NOT NULL, 
  Validator  varchar(255) NOT NULL, 
  Expiration timestamp NOT NULL, 
  Type       int4 NOT NULL, 
  PRIMARY KEY (ID));
ALTER TABLE AuthToken ADD CONSTRAINT AgentToken FOREIGN KEY (AgentID) REFERENCES Agent (ID);

The AuthServiceImpl will then be extended to issue, persist and validate a token based on the guidelines of the Paragon Initiative Enterprises Blog and the OWASP:

class AuthServiceImpl implements AuthService {
    public function issueToken($type = self::AGENT_TOKEN, $email = null) {
        $token = new AuthToken();
        $token->setSelector(bin2hex(random_bytes(5)));
        $token->setType(self::AGENT_TOKEN);
        $token->setAgentid($this->currentAgentId);
        $timestamp = (new \DateTime('now'))->modify('+30 days');
        $token->setExpiration($timestamp->format("Y-m-d H:i:s"));
        $validator = random_bytes(20);
        $token->setValidator(hash('sha384', $validator));
        $authTokenDAO = new AuthTokenDAO();
        $authTokenDAO->create($token);
        return $token->getSelector() .":". bin2hex($validator);
    }

    public function validateToken($token) {
        $tokenArray = explode(":", $token);
        $authTokenDAO = new AuthTokenDAO();
        $authToken = $authTokenDAO->findBySelector($tokenArray[0]);
        if (!empty($authToken)) {
            if(time()<=(new \DateTime($authToken->getExpiration()))->getTimestamp()){
                if (hash_equals(hash('sha384', hex2bin($tokenArray[1])), $authToken->getValidator())) {
                    $this->currentAgentId = $authToken->getAgentid();
                    return true;
                }
            }
            $authTokenDAO->delete($authToken);
        }
        return false;
    }
}

The token, which is managed by the AuthToken domain object (as shown in the Domain Model), is separated into a selector and a validator. The selector is a random key to select a token from the database. This selector is not hashed and not the user id to avoid exposing the number of users of the system. The validator is a random key that must be hashed before storing in the database.

The AuthController will then be extended to set a remember me cookie containing the issued token:

public static function login(){
    $authService = AuthServiceImpl::getInstance();
    if($authService->verifyAgent($_POST["email"],$_POST["password"]))
    {
        $token = $authService->issueToken();
        $_SESSION["agentLogin"]["token"] = $token;
        if(isset($_POST["remember"])) {
            setcookie("token", $token, (new \DateTime('now'))->modify('+30 days')->getTimestamp(), "/", "", false, true);
        }
    }
}

Finally, the remember me feature can be added to the login view:

<div class="form-group">
    <div class="checkbox">
        <label class="control-label">
            <input type="checkbox" name="remember" />Remember me for 30 days</label>
    </div>
</div>

Stage 13: Email and Password Reset

In stage 13, an email sending and password reset functionalities will be implemented.

Email

Although there exists a mail functionality in PHP, an external mail service/API is used here. An external API can offer additional functionalities and can provide an easier implementation. In this reference project, the SendGrid API is used for sending emails. Before using the SendGrid API an account and an API-Key is required. This API key must be stored in an external configuration file and kept outside of version control. It is advisable that the API key will be stored in the .env Config File as follows:

[email]
email.sendgrid-apikey=

If running this reference project on Heroku, the SendGrid API Key should be stored in an environment variable, e.g. SENDGRID_APIKEY:

if (isset($_ENV["SENDGRID_APIKEY"])) {
    self::$config["email.sendgrid-apikey"] = $_ENV["SENDGRID_APIKEY"];
}

Based on the SendGrid API documentation the email service is implemented using plain PHP as follows:

class EmailServiceClient
{

    public static function sendEmail($toEmail, $subject, $htmlData){
        $jsonObj = self::createEmailJSONObj();
        $jsonObj->personalizations[0]->to[0]->email = $toEmail;
        $jsonObj->subject = $subject;
        $jsonObj->content[0]->value = $htmlData;

        $options = ["http" => [
            "method" => "POST",
            "header" => ["Content-Type: application/json",
                "Authorization: Bearer ".Config::get("email.sendgrid-apikey").""],
            "content" => json_encode($jsonObj)
        ]];
        $context = stream_context_create($options);
        $response = file_get_contents("https://api.sendgrid.com/v3/mail/send", false, $context);
        if(strpos($http_response_header[0],"202"))
            return true;
        return false;
    }

    protected static function createEmailJSONObj(){
        return json_decode('{
          "personalizations": [
            {
              "to": [
                {
                  "email": "email"
                }
              ]
            }
          ],
          "from": {
            "email": "[email protected]",
            "name": "WE-CRM"
          },
          "subject": "subject",
          "content": [
            {
              "type": "text/html",
              "value": "value"
            }
          ]
        }');
    }
}

Password Reset

A password reset functionality without a two-factor authentication can't be implemented securely since email is always the weak point. Nevertheless, it is strongly recommended to implement a token-based password reset functionality similar to the Remember Me implementation.

The AuthServiceImpl has been extended to issue RESET_TOKEN:

class AuthServiceImpl implements AuthService {

//...
    
    public function issueToken($type = self::AGENT_TOKEN, $email = null) {
        $token = new AuthToken();
        $token->setSelector(bin2hex(random_bytes(5)));
        if($type===self::AGENT_TOKEN) {
            $token->setType(self::AGENT_TOKEN);
            $token->setAgentid($this->currentAgentId);
            $timestamp = (new \DateTime('now'))->modify('+30 days');
        }
        elseif(isset($email)){
            $token->setType(self::RESET_TOKEN);
            $token->setAgentid((new AgentDAO())->findByEmail($email)->getId());
            $timestamp = (new \DateTime('now'))->modify('+1 hour');
        }else{
            throw new HTTPException(HTTPStatusCode::HTTP_406_NOT_ACCEPTABLE, 'RESET_TOKEN without email');
        }
        $token->setExpiration($timestamp->format("Y-m-d H:i:s"));
        $validator = random_bytes(20);
        $token->setValidator(hash('sha384', $validator));
        $authTokenDAO = new AuthTokenDAO();
        $authTokenDAO->create($token);
        return $token->getSelector() .":". bin2hex($validator);
    }
}

The AuthServiceImpl and the EmailServiceClient can then be used in the AgentPasswordResetController to provide the password-reset functionality:

class AgentPasswordResetController
{

    public static function resetView(){
        $resetView = new TemplateView("agentPasswordReset.php");
        $resetView->token = $_GET["token"];
        echo $resetView->render();
    }
    
    public static function requestView(){
        echo (new TemplateView("agentPasswordResetRequest.php"))->render();
    }
    
    public static function reset(){
        if(AuthServiceImpl::getInstance()->validateToken($_POST["token"])){
            $agent = AuthServiceImpl::getInstance()->readAgent();
            $agent->setPassword($_POST["password"]);
            $agentValidator = new AgentValidator($agent);
            if($agentValidator->isValid()){
                if(AuthServiceImpl::getInstance()->editAgent($agent->getName(),$agent->getEmail(), $agent->getPassword())){
                    return true;
                }
            }
            $agent->setPassword("");
            $resetView = new TemplateView("agentPasswordReset.php");
            $resetView->token = $_POST["token"];
            echo $resetView->render();
            return false;
        }
        return false;
    }

    public static function resetEmail(){
        $token = AuthServiceImpl::getInstance()->issueToken(AuthServiceImpl::RESET_TOKEN, $_POST["email"]);
        $emailView = new TemplateView("agentPasswordResetEmail.php");
        $emailView->resetLink = $GLOBALS["ROOT_URL"] . "/password/reset?token=" . $token;
        return EmailServiceClient::sendEmail($_POST["email"], "Password Reset Email", $emailView->render());
    }

}

Stage 14: PDF

In this stage 14, a PDF creation service is used, which is called HyPDF. HyPDF provides an API for generating PDF files from text or HTML.

Similar to stage 13, the HyPDF username and password must be stored outside of the source code, preferably in a config file:

[pdf]
pdf.hypdf-user=
pdf.hypdf-password=

If running this reference project on Heroku, the HyPDF username and password should be stored in an environment variable:

if (isset($_ENV["HYPDF_USER"])) {
    self::$config["pdf.hypdf-user"] = $_ENV["HYPDF_USER"];
}
if (isset($_ENV["HYPDF_PASSWORD"])) {
    self::$config["pdf.hypdf-password"] = $_ENV["HYPDF_PASSWORD"];
}

To generate a PDF out of an HTML file, the HyPDF API can be used as follows:

class PDFServiceClient
{

    public static function sendPDF($htmlData){
        $jsonObj = self::createPDFJSONObj();
        $jsonObj->user = Config::get("pdf.hypdf-user");
        $jsonObj->password = Config::get("pdf.hypdf-password");
        $jsonObj->content = $htmlData;

        $options = ["http" => [
            "method" => "POST",
            "header" => ["Content-Type: application/json"],
            "content" => json_encode($jsonObj)
        ]];
        $context = stream_context_create($options);
        $response = file_get_contents("https://www.hypdf.com/htmltopdf", false, $context);
        if(strpos($http_response_header[0],"200"))
            return $response;
        return false;
    }

    protected static function createPDFJSONObj(){
        return json_decode('{"content": "HTML", "user": "HYPDF_USER", "password": "YOUR_HYPDF_PASSWORD", "test": "true"}');
    }
}

The return value of the sendPDF method contains a PDF, which can be echoed back to the browser.

Stage 15: REST Service API

In this stage 15, an own REST Service API is provided and implemented.

API Model

API Authorization

First, the API authorization is implemented in the ServiceEndpoint using basic authentication with username and password at a first stage to retrieve an access token, and then a token based authorization for the subsequent API calls:

class ServiceEndpoint
{

    public static function authenticateToken(){
        if (isset($_SERVER["HTTP_AUTHORIZATION"])){
            if(strripos($_SERVER["HTTP_AUTHORIZATION"], " ")){
                list($type, $data) = explode(" ", $_SERVER["HTTP_AUTHORIZATION"], 2);
                if (strcasecmp($type, "Bearer") == 0) {
                    if(AuthServiceImpl::getInstance()->validateToken($data)) {
                        return true;
                    }
                }
            }
        }
        return false;
    }

    public static function authenticateBasic(){
        if (isset($_SERVER["HTTP_AUTHORIZATION"])){
            if(strripos($_SERVER["HTTP_AUTHORIZATION"], " ")) {
                list($type, $data) = explode(" ", $_SERVER["HTTP_AUTHORIZATION"], 2);
                if (strcasecmp($type, "Basic") == 0) {
                    list($name, $password) = explode(':', base64_decode($data));
                    if (AuthServiceImpl::getInstance()->verifyAgent($name, $password)) {
                        return true;
                    }
                }
            }
        }
        return false;
    }
    //...
}

JSON Serialization

A serialization and deserialization mechanism is required map the Domain Objects with JSON objects. This reference project provides an abstract data transfer object (DTO), which provides this serialisation mechanism:

abstract class AbstractJSONDTO implements \JsonSerializable
{
    //...

    public static function Deserialize($decodedJSON)
    {
        $className = get_called_class();
        $classInstance = new $className();

        foreach ($decodedJSON as $key => $value) {
            if (property_exists($classInstance, $key)) {
                $classInstance->{$key} = $value;
            }
        }

        return $classInstance;
    }

    function jsonSerialize()
    {
        return get_object_vars($this);
    }
}

Service Endpoint

Based on the API Model, a ServiceEndpint on business layer has been implemented as follows:

class ServiceEndpoint
{

    //...

    public static function loginBasicToken(){
        $authService = AuthServiceImpl::getInstance();
        HTTPHeader::setHeader("Authorization: " . $authService->issueToken(), HTTPStatusCode::HTTP_204_NO_CONTENT, false);
    }

    public static function validateToken(){
        HTTPHeader::setStatusHeader(HTTPStatusCode::HTTP_202_ACCEPTED);
    }

    public static function findAllCustomer(){
        $responseData = (new CustomerServiceImpl())->findAllCustomer();
        HTTPHeader::setHeader("Content-Type: application/json", HTTPStatusCode::HTTP_200_OK, true);
        echo json_encode($responseData);
    }

    public static function readCustomer($id){
        $responseData = (new CustomerServiceImpl())->readCustomer($id);
        HTTPHeader::setHeader("Content-Type: application/json", HTTPStatusCode::HTTP_200_OK, true);
        echo json_encode($responseData);
    }

    public static function updateCustomer($customerId = null){
        $requestData = json_decode(file_get_contents("php://input"), true);
        $customer = Customer::Deserialize($requestData);
        $customerValidator = new CustomerValidator($customer);
        if($customerValidator->isValid()) {
            if (is_null($customerId)) {
                $customer = (new CustomerServiceImpl())->createCustomer($customer);
                $location = $GLOBALS["ROOT_URL"] . $_SERVER['PATH_INFO'] . $customer->getId();
                HTTPHeader::setHeader("Location: " . $location, HTTPStatusCode::HTTP_201_CREATED, true);
            } else {
                $customer->setId($customerId);
                (new CustomerServiceImpl())->updateCustomer($customer);
                HTTPHeader::setStatusHeader(HTTPStatusCode::HTTP_204_NO_CONTENT);
            }
        }
        else{
            return false;
        }
        return true;
    }

    public static function createCustomer(){
        return self::updateCustomer();
    }

    public static function deleteCustomer($id){
        (new CustomerServiceImpl())->deleteCustomer($id);
        HTTPHeader::setStatusHeader(HTTPStatusCode::HTTP_204_NO_CONTENT);
    }

}

API Routes

In most PHP and node.js frameworks, it is common to keep the API routes separated from the ServiceEndpoint implementation. In the following route definitions the API Model has been realized:

Router::route_auth("GET", "/api/token", $authAPIBasicFunction, function () {
    ServiceEndpoint::loginBasicToken();
});

Router::route_auth("HEAD", "/api/token", $authAPITokenFunction, function () {
    ServiceEndpoint::validateToken();
});

Router::route_auth("GET", "/api/customer", $authAPITokenFunction, function () {
    ServiceEndpoint::findAllCustomer();
});

Router::route_auth("GET", "/api/customer/{id}", $authAPITokenFunction, function ($id) {
    ServiceEndpoint::readCustomer($id);
});

Router::route_auth("PUT", "/api/customer/{id}", $authAPITokenFunction, function ($id) {
    ServiceEndpoint::updateCustomer($id);
});

Router::route_auth("POST", "/api/customer", $authAPITokenFunction, function () {
    ServiceEndpoint::createCustomer();
});

Router::route_auth("DELETE", "/api/customer/{id}", $authAPITokenFunction, function ($id) {
    ServiceEndpoint::deleteCustomer($id);
});

Stage 16: JavaScript and jQuery Client

In stage 16, a JavaScirpt/jQuery based API consumer is implemented. The implementation is realized by AJAX calls provided as functions in an app.js file:

<script src="assets/js/jquery.min.js"></script>
<script src="assets/js/app.js"></script>

API Login and Local Token Storage

On every HTML view, a token validation will be performed:

<script language="JavaScript">
    validateToken(function (result) {
        if (!result) {
            window.location.replace("agentLogin.html");
        }
    });
</script>

If the current token in the local storage is not available or invalid, the user is asked to provide email and password, which then performs a basic authorization to receive a valid access token:

function getToken(email, password, callback) {
    $.ajax({
        type: "GET",
        headers: {
            "Authorization": "Basic " + btoa(email + ":" + password)
        },
        url: serviceEndpointURL + "/token",
        success: function (data, textStatus, response) {
            localStorage.setItem("token", response.getResponseHeader("Authorization"));
            callback(true);
        },
        error: function (jqXHR, textStatus, errorThrown) {
            console.log(jqXHR, textStatus, errorThrown);
            callback(false);
        }
    });
}

function validateToken(callback) {
    $.ajax({
        type: "HEAD",
        headers: {
            "Authorization": "Bearer " + localStorage.getItem("token")
        },
        url: serviceEndpointURL + "/token",
        success: function (data, textStatus, response) {
            callback(true);
        },
        error: function (jqXHR, textStatus, errorThrown) {
            callback(false);
        }
    });
}

AJAX Calls for API Consumption

It is not required to use third-party libraries for REST clients. Nevertheless, libraries, such as jQuery, can enhance the readability of REST calls significantly. REST APIs can be consumed with jQuery based AJAX calls using the following pattern:

function getCustomers(callback) {
    $.ajax({
        type: "GET",
        headers: {
            "Authorization": "Bearer " + localStorage.getItem("token")
        },
        dataType: "json",
        url: serviceEndpointURL + "/customer",
        success: function (data, textStatus, response) {
            console.log(textStatus);
            callback(data);
        },
        error: function (jqXHR, textStatus, errorThrown) {
            console.log(jqXHR, textStatus, errorThrown);
        }
    });
}

function postCustomer(customer, callback) {
    $.ajax({
        type: "POST",
        headers: {
            "Authorization": "Bearer " + localStorage.getItem("token")
        },
        url: serviceEndpointURL + "/customer",
        data: customer,
        success: function (data, textStatus, response) {
            console.log(textStatus);
            callback(data);
        },
        error: function (jqXHR, textStatus, errorThrown) {
            console.log(jqXHR, textStatus, errorThrown);
        }
    });
}

function getCustomer(customerID, callback) {
    $.ajax({
        type: "GET",
        headers: {
            "Authorization": "Bearer " + localStorage.getItem("token")
        },
        dataType: "json",
        url: serviceEndpointURL + "/customer/" + customerID,
        success: function (data, textStatus, response) {
            console.log(textStatus);
            callback(data);
        },
        error: function (jqXHR, textStatus, errorThrown) {
            console.log(jqXHR, textStatus, errorThrown);
        }
    });
}

This reference project uses plain jQuery to append HTML elements to the DOM:

function loadData() {
    getCustomers(function (result) {
        $("#tableData").empty();
        $.each(result, function (i, item) {
            $("#tableData").append($("<tr>").append($("<td>").text(item.id)).append($("<td>").text(item.name)).append($("<td>").text(item.email)).append($("<td>").text(item.mobile)).append($(
                "<td><div class='btn-group btn-group-sm' role='group'>" +
                "<a class='btn btn-default' role='button' href='customerEdit.html?id=" + item.id + "'> <i class='fa fa-edit'></i></a>" +
                "<button class='btn btn-default' type='button' data-target='#confirm-modal' data-toggle='modal' data-id='" + item.id + "'> <i class='glyphicon glyphicon-trash'></i></button>" +
                "</div></td></tr>"
            )));
        });
    });
}

Don't forget that .append() can be dangerous when appending user input or database data. This untrusted data could contain harmful XSS-related JavaScript. Therefore untrusted data must be added using .text() to escape all special characters.

Nevertheless, it might be advisable for readability to use a third-party library or a jQuery extension for appending HTML to the DOM.

Deployment

Project Set-Up

Visual Paradigm

Default Parameter Direction Configuration

Window -> Project Options

Postgresql Database Generation

Git

The project contains a .gitignore file to keep certain

Heroku Deployment

Deploy

Maintainer

License

About

This is a reference project elaborated by the students step-by-step in every FHNW web engineering lecture.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published