Skip to main content

Bad Cakephp habbits

Bad CakePHP Habits & How to Rectify Them!

SUNDAY OCTOBER 27, 08:23

I've been thinking a lot recently about how I can best help the CakePHP community - to try to give back some of the love which I have received over the years.
I have been fortunate to use CakePHP for over 5 years now, and in that time have gotten to know some key members of the CakePHP community.
So, what I wanted to go over was a few bad practices which I have seen over the years and how the correct procedure should be implemented.  This isn't to say my code is perfect, but as a programmer we are always learning, so it is important to follow the best practices and adjust to them as you learn them!

CAKEPHP CONVENTIONS

There are actually CakePHP coding conventions which should be followed.  These can be viewed here.  I will highlight a few things which I often notice when viewing other programmers code.
1) Control Structures.  So often you see programmers get this wrong, and in even some cases bring practices for other coding languages.  CakePHP expects the following syntax:
if ((expr_1) || (expr_2)) {
    // action_1;
} elseif (!(expr_3) && (expr_4)) {
    // action_2;
} else {
    // default_action;
}

In the control structures there should be 1 (one) space before the first parenthesis and 1 (one) space between the last parenthesis and the opening bracket.  So this means that the following is incorrect:
if($this->request->data){

}
Note the spacing between the if and ( and the ) and {
Always use curly brackets in control structures, even if they are not needed. They increase the readability of the code, and they give you fewer logical errors.
So for example the following is incorrect:
if ($foo)
    $bar = true
This should be formatted like this
if ($foo) {
    $bar = true
}
Finally, watch where the brackets go.  Brackets shouldn't go on new lines, and make sure all your brackets line up.  Make sure the code is readable and indented correctly, so that each new bracket is inline with the closing bracket.
So, here are some incorrect examples which I see often:
if ($foo)
{
    $bar = true;
}
This is incorrect, the opening bracket should be on the first line:
if ($foo) {    
    $bar = true;
if ($action) {
    $to = false;
}
}
The indentation needs to line up correctly.
I often hear programmers say "but I am too busy to make the code neat...." My response is - "trust me, neat code will stand the test of time".  Writing code which isn't readable will be a nightmare to come back to if you need to make a change in a few months.

CONTAINABLE & RECURSIVE

I was fortunate recently to have an informal discussion with a database developer from Facebook.  We started talking about CakePHP and he said to me, "oh, that uses ORM doesn't it, that can be scary."  I asked him what he meant, and he commented that with Object-relational mapping (ORM) it is easy for SQL queries to become unnecessarily large.
And he is right in a way.  Part of CakePHP's magic is in it's use of ORM and the way that it groups different database table relations together.  By default, CakePHP automatically selects any related 'Belongs To', 'Has One' and 'Has Many' data, and this can lead to very large SQL queries.  These queries may not be a concern when you are initially developing an application, but after 6 months of collecting live data, and you may find the application becomes very slow, and in some cases crashes if the queries aren't optimised.
I look out for two things when auditing an existing website.  Firstly, has the default recursive level been changed.  By default, CakePHP sets the recursive level to 1, which in my opinion is too high.  I always set it to -1 and then use the containable behaviour to get any related models.
And that leads nicely to the second thing I look for - has the Containable Behaviour been used.  I often have new clients come to me who say that CakePHP is slow, and the reason is almost always because Containable hasn't been used!  A good programmer will optimise their SQL queries regardless of how much "auto-magic" is done behind the scenes.
The containable behavior wasn't added until CakePHP 1.2, but boy has it made a difference.  Be sure to use containable as much as possible, as it is such an effective way to optimise your SQL.  For more information on how to implement and use the Containable behavior, click here.

FAT MODELS, SKINNY CONTROLLERS

Good CakePHP code will have the logic in the model files.  This takes a bit to get used to, but once mastered there is no looking back!  A controller file should be used for what it is intended for in the MVC pattern - controlling!  So use your controller file to handle user actions, while let the code logic go in the model file.
A good example of this would be a simple CRUD - an everyday action!  Lets take the add posts function from the blog tutorial as an example.  The default add function is as follows:
public function add() {
    if ($this->request->is('post')) {
        $this->Post->create();
        if ($this->Post->save($this->request->data)) {
            $this->Session->setFlash(__('Your post has been saved.'));
            return $this->redirect(array('action' => 'index'));
        }
        $this->Session->setFlash(__('Unable to add your post.'));
    }
}
This controller action is fine for a simple add, but what would happen if you wanted to do things such as send an email to the admin when a post was added, or update another model association when a post was added.  This is additional logic, but this logic shouldn't go into our controller file.
Instead we would write a function for this in our Post.php model, perhaps something like this:
public function addPost($data = array(), $emailAdmin = true) {
    $this->create();
    $this->save($data);

    // update any other tables

    // send the email to the admin user
    if ($emailAdmin) {
    
    }

    // if all is successful
    return true;
}
This would then result in a small change to the controller action as follows:
public function add() {
    if ($this->request->is('post')) {
        if ($this->Post->addPost($this->request->data)) {
            $this->Session->setFlash(__('Your post has been saved.'));
            return $this->redirect(array('action' => 'index'));
        }
        $this->Session->setFlash(__('Unable to add your post.'));
    }
}
As you can see, the new action is actually one less line, because the $this->Post->create() has been moved to the model file.
This is a perfect, everyday example of where moving logic to the model file is a good idea - and it certainly makes for a much cleaner code base!

RETURN OFTEN, RETURN EARLY

This is always a bit of an ongoing debate, but returning often, and returning early certainly does make for much cleaner looking code.  This applies to the model methods more than anything else.
But what exactly do I mean?  Well, lets take a look at the method we added above:
public function addPost($data = array(), $emailAdmin = true) {
    $this->create();
    $this->save($data);

    // update any other tables

    // send the email to the admin user
    if ($emailAdmin) {
    
    }

    // if all is successful
    return true;
}
To return often, and return early means that as we run through our function, we check to make sure the everything is OK on a regular basis.  If it isn't, then we return false, or return an error.
It might be easiest to show this with an example!  There are two ways the above function could be written:
public function addPost($data = array(), $emailAdmin = true) {
    if ($data) {
        $this->create();
        $result = $this->save($data);

        if ($result) {
            // update any other tables

            // send the email to the admin user
            if ($emailAdmin) {
                // send the admin email
            }
        } else {
            // problem saving the data
            return false;
        }

        // if all is successful
        return true;
    } else {
        // no data submitted
        return false;
    }
}
See how the code quickly becomes unreadable?  There are if and else's all over the place, and the function quickly becomes one big indentation.  Don't get me wrong, I love clean indentation, but watch how the function looks if it written with the return often, return early principle.
public function addPost($data = array(), $emailAdmin = true) {
    if (!$data) {
        // no data submitted
        return false;
    }

    $this->create();
    $result = $this->save($data);

    if (!$result) {
        // problem saving the data
        return false;
    }

    // update any other tables

    // send the email to the admin user
    if ($emailAdmin) {
        // send the admin email
    }

    // if all is successful
    return true;
}
Straight away, in this small example, you will now see the code only has a single indentation and is much more readable.  The logic actually makes more sense - let the logic run through line by line, and if there is any issue along the way, return the error and don't proceed to the next line.
This allows a programmer to write the same way that we read - reading code from left to right, top to bottom, rather than in different blocks, which can quickly get confusing!

DRY

DRY stands for Don't Repeat Yourself, and it is a philosophy which should be followed when coding in CakePHP.  With object-oriented code, there is no excuse for repeating the same block of code twice!
Here is a few tips to ensure you don't repeat yourself:
1) As mentioned above, aim to put logic in model files so you can share the logic.
2) In your view files, if you are repeating views, either create the view code as an Element, or even a custom helper.
3) Set up some configuration settings - the app/Config/bootstrap.php file is a great place for this.  So that you aren't hard coding things like the application name and the main email address.  The last thing you want to do is go through hundreds of files just because the client has asked to update an email address in an application.
4) Always ask yourself - if I am repeating code, is there a better way to write this code, and am I putting this code in the right place!  Chances are, if you need to repeat code, it could be written better.

DOC BLOCKING & INLINE COMMENTS

The last point I will make is in regards to comments. Firstly, doc blocking.  A Doc Block is when you document a method or an action.  It takes only a minute to record a little about what a function is doing, but it makes such a difference in terms of readability of code.
With CakePHP Doc Blocks, they need to go against the left margin of the page.  So a simple example if to use the code from above.
/**
 * Adds & saves a post as well as emails the admin to let them know the post has been added.
 * Also performs some saving to another table
 *
 * @param   array $data The post data
 * @param   bool $emailAdmin If set to true, will email the website admin
 * @return  bool Returns true if successful
 */
    public function addPost($data = array(), $emailAdmin = true) {

As you will see, it doesn't take long to write a doc block, but it makes a huge difference in terms of longevity of the code.  Ultimately, it means the code can live on past you as the developer.
Likewise with in-line comments.  Don't be scared to explain what your code is doing and why!  It makes it a lot easier in the long run to understand your code, especially if another developer is looking at it!

Comments

Popular posts from this blog

How to Create a PDF file in Cakephp 2.0 using Fpdf

Step 1: Download FPDF folder from  http://www.fpdf.org/  . Step 2: Unzip the downloaded Fpdf file and name it “fpdf” or however you require and make sure that you use the same name while calling it. Step 3: Move the “fpdf” unzipped files to  your /app/Vendor directory within Cakephp. Now you should have the directory path as   /app/Vendor/fpdf. Step 4: Create a new Cakephp layout file for the pdfs. We will use this layout when serving a pdf to the client. Create a file called pdf.ctp inside of /app/View/Layouts. Add the following code to /app/View/Layouts/pdf.ctp Layout: /app/View/Layouts/pdf.ctp 1 2 3 4 <?php      header ( 'Content-Disposition: attachment; filename="downloaded.pdf"' ) ;      echo $content_for_layout ; ?> The header function above tells the browser that it is going to receive...

Setup CakePHP Using Xampp On Windows

Step 1: Download XAMPP  and  CakePHP .   Step 2: Install Xampp Once you have installed Xampp (version 1.7.3) on your Windows with the default option, all your files will be located in the C:\xampp folder. Step 3: Mod Rewrite Module Once Xampp is installed as the local server, you can then proceed to enable mod_rewrite. To do so, you will have to open the httpd.conf file that is located in C:\xampp\apache\conf and uncomment by removing # from the following line: # LoadModule rewrite_module modules/mod_rewrite.so Step 4: Place CakePHP Files in a New Folder Extract the CakePHP (version 1.3.8) zip file and copy all its contents to your local web server, which in this instance is C:\xampp\htdocs\cakephp . I have decided to name the CakePHP folder as cakephp, and in it, you will find many files and folders for the framework, including app, cake, docs, vendors, .htaccess, and index.php. Step 5: Set Up Virtual Host Open the httpd-vhosts.conf file from th...

Dynamic Sitemap Generation plugin in Cakephp

Here for the SEO implementation we need to generate the sitemap.xml in our application which is accessed by the webmaster tool. So here i am outlined the steps to generate the Xml file . 1. Lets think we have controller by name sitemap,Inside that create an action by name sitemap and paste the following code    public function sitemap(){     $this->layout='ajax';     $this->RequestHandler->respondAs('xml');     $listData = $this->Sitemap->find('all',/*Conditions if you have any*/);     $this->set(compact('listData')); } I through variable $listData to render all data( Keywords,Title,Url,etc ...) that will be shown in sitemap.xml.  This   depends   on the   dynamic link  what   we want to   show  in sitemap.xml.For request handler to work include the RequestHandler component public $components = array('RequestHandler'); Step 2. Create View based on...

Druapl 8 Folder and File Permissions

Drupal direcories: find /var/www/html -type d -exec chmod 755 {} \; Drupal files: find /var/www/html -type f -exec chmod 644 {} \; Drupal all content related directories: find /var/www/html/your_site_name/ sites/b2b_cms_mi/files -type d -exec chmod 775 {} \; Drupal all content related files: find /var/www/html/your_project_folder/ sites/default/files -type f -exec chmod 664 {} \; Drupal content folder permission: chmod 777 /var/www/html/your_project_folder/ sites/default/files .htaccess permission cd /var/www/html/your_project_folder/ sites/default/files/config_ fdwgNYChzdnUSBU8hEm-QEe3_ SXdb6kzCJ3ffEC4sB7njFXuK06hHUF OJE888-GN0_gpEE1b0w/sync chmod 777 .htaccess cd /var/www/html/your_project_folder/ sites/default/files/ chmod 777 .htaccess Note:-  your_project_folder is the name of the project. default can be a site folder name, if the application is multisite or else it will be a default folder.