CREATE TABLE notes ( id INT UNSIGNED AUTO_INCREMENT PRIMARY KEY, title VARCHAR(50), body TEXT, created DATETIME DEFAULT NULL, modified DATETIME DEFAULT NULL );
Cake is a rapid development framework for PHP which uses commonly known design patterns like ActiveRecord, Association Data Mapping, Front Controller and MVC. Our primary goal is to provide a structured framework that enables PHP users at all levels to rapidly develop robust web applications, without any loss to flexibility.
Hot Features:
Model, View, Controller Architecture
View Helpers for AJAX, Javascript, HTML Forms and more
Built-in Validation
Application Scaffolding
Application and CRUD code generation via Bake
Access Control Lists
Data Sanitization
Security, Session, and Request Handling Components
Flexible View Caching
And More…
Active, Friendly Community - Just join our IRC channel to see who's in. We'd love to help you get started.
Flexible License - Cake is distributed under the MIT License
Clean IP - Every line of code was written by the CakePHP development team
Extremely Simple - Just look at the name…It's Cake
Rapid Development - Build apps faster than ever before (check out the zZine article)
Best Practices - Cake is easy to understand and sets the industry standard in security authentication, and session handling, among other features.
OO - Whether you are a seasoned object-oriented programmer or a beginner, you'll feel comfortable
No Configuration - Set-up the database and watch the magic begin
by Fabio Cevasco
PHP has its weak spots of course, among which a sort of inherent disorganization both at core level (no namespaces, inconsistent naming conventions, etc.) and at user lavel: PHP code can be seamlessly integrated into HTML pages, and a lot of developers - newbies but also seasoned ones - have been using this feature to create the most tangled and messy examples of spaghetti code.
Indeed using a single php file for connecting to a database, fetching some results, doing some operations with them and then displaying them via a HTML table just works but this is not sufficient to make it a good coding practice. Imagine a lot of huge, monolitic scripts each including n others almost equally complex and tangled: the picture you get is just some very disorganized code which still works and maybe is even fast to run, but it's a real nightmare to maintain and debug.
One thing PHP code developers and Zend in particular didn't do until recently was try to bring order to the chaos and create an official framework offering some sort of structure and conventions to make PHP application easier to maintain and develop. It seems that finally something is happening with the Zend framework but perhaps is too late already: the new Zend framework is still under development while there are a lot of various alternatives out there which are stable enough to do wonders, and "CakePHP" is undoubtedly one of the most valid.
CakePHP was originally born as an attempt to port the architecture and functionality offered by "Ruby on Rails":http://www.rubyonrails.com/ to PHP, something which various other frameworks have tried to do in different ways. Although Cake has many similarities with the popular Ruby framework, it's currently evolving independently: it's not an actual port - that would be close to impossible, considering the features offered by Ruby itself which are not present in PHP - but rather a very PHP-specific implementation of Rails' architecture and RAD philosophy. CakePHP separates itself from the rest of the PHP world by concentrating on simplicity, portability, flexibility and scalability. CakePHP does not require any additional libraries like PEAR or Propel. CakePHP includes an architecture that allows for easily extending its many built-in features through components and helpers. All this in under 300K.
The only really important concept to grasp is its MVC (Model-View-Controller) architecture, which is actually common to many other PHP frameworks - including, recently, the Zend's. If you're new to this very intuitive and yet powerful concept, imagine to logically divide your web application in a few major sections (controllers), like a blog, a photo gallery and so on. Each section will obviously have access to one or more tables in your database (models) and will also include some template files (views) used to present content to the users. This may be a quite informal but concrete representation of the MVC architecture, using a real-world example. In other words, controller files handle your application's business logic, keeping it separate from the code used to represent the datasource (model) and the presentation layer (view). This separation may seem a bit too strict at first - you may be tempted to try including some business logic in your views, for example - but will be able to increase your code maintainability in the long term.
CakePHP can be freely downloaded from the official site and requires no installation at all: just extract the content of the compressed archive file into a public directory of your webserver to deploy the framework and your project's skeleton. The directory structure of CakePHP is designed to be easy to understand and well defined for intetlligent code seperation:
app/
config/
controllers/
models/
plugins/
tmp/
vendors/
views/
webroot/
cake/
config/
docs/
libs/
vendors/
To an extent, even someone who is not familiar with CakePHP can understand roughly where each particular file should be placed. There are three first level folders: cake containing the framework itself, app which hosts an empty project and vendors which can be used for third-parties libraries. It's worth noting that when creating a project with CakePHP you'll never have to touch the contents of the cake folder: the framework's source code is always kept separated by the user project. One of the most important part of Rails' philosophy Cake ported to PHP is convention over configuration, in fact there's no need for dozens of configuration files or no hidden settings to configure: the framework knows where everything is and it's smart enough to load the appropriate classes depending on a particular task, and of course a fixed directory structure plays a key role for this smart behaviour.
There are only two configuration files which can be modified, one is app/config/core.php and contains various application settings like the debug level, session timeout, etc., and the other - app/config/database.php - is necessary to establish a (persistent) database connection:
The "15 minute blog tutorial":http://manual.cakephp.org/chapter/18 is the best way to learn "how to bake". After trying it out yourself you'll be able to grasp all CakePHP's basic concepts and start coding your own application.
By Fabio Cevasco July 12th 2006
Readers who already know Ruby on Rails may find CakePHP very similar to it. For one thing, Cake is based on an MVC-like architecture that is both powerful and easy to grasp: controllers, models and views guarantee a strict but natural separation of business logic from data and presentation layers.
Controllers contain the logic of your application. Each controller can offer different functionality; controllers retrieve and modify data by accessing database tables through models; and they register variables and objects, which can be used in views.
Models are active representations of database tables: they can connect to your database, query it (if instructed to do so by a controller) and save data to the database. It is important to note that in order to correctly apply the MVC architecture, there must be no interaction between models and views: all the logic is handled by controllers.
Views can be described as template files that present their content to the user: variables, arrays [9] and objects that are used in views are registered through a controller. Views should not contain complex business logic; only the elementary control structures necessary to perform particular operations, such as the iteration of collected data through a foreach construct, should be contained within a view.
This architecture can greatly improve the maintainability and the organization of your site's code:
It separates business logic from presentation and data retrieval.
A site is divided into logical sections, each governed by a particular controller.
When testing and debugging an application, any developer accustomed to CakePHP's structure will be able to locate and correct errors without knowing all of the details of the code.
Controllers, models and views are stored in pre-defined directories within CakePHP's directory structure. Here's the directory structure that's used:
app/
config/
controllers/
models/
plugins/
tmp/
vendors/
views/
webroot/
cake/
config/
docs/
libs/
vendors/
This directory scheme must be preserved, as it is essential if the framework itself is to work. Cake, like Rails, believes in the importance of convention over configuration: in order to deploy an application, rather than modify dozens of different configuration files, it's important only to place everything in its proper place; then, you can let the framework do the rest.
As an example, let's now look at building a simple memo application that allows you to add, edit, display and delete personal notes, which are stored in a MySQL [10] database.
We need to create a new MySQL database named memo that's accessible by a user named memouser, and a new table named notes:
CREATE TABLE notes ( id INT UNSIGNED AUTO_INCREMENT PRIMARY KEY, title VARCHAR(50), body TEXT, created DATETIME DEFAULT NULL, modified DATETIME DEFAULT NULL );
Note how the table uses the plural notes. Now, edit your database configuration file (/app/config/database.php.default) as follows, and save it as database.php (yes: the name matters!):
<?php class DATABASE_CONFIG { var $default = array [17]('driver' => 'mysql', 'connect' => 'mysql_pconnect', 'host' => 'localhost', 'login' => 'memouser', 'password' => 'userpassword', 'database' => 'memo' ); var $test = array('driver' => 'mysql', 'connect' => 'mysql_pconnect', 'host' => 'localhost', 'login' => 'user', 'password' => 'password', 'database' => 'project_name-test'); } ?>
If you refresh the default page, CakePHP will notify you that the database is now accessible.
CakePHP is now configured properly, and we can turn to the development of our application. The framework offers a useful Rails-inspired feature called scaffolding, which basically allows the creation of an interface that's able to perform Create, Read, Update and Delete (CRUD) database operations with only a few lines of code. This is particularly useful when you want a particular area of your application to be available for testing purposes quickly, and you don't want to spend time coding it properly — yet.
To create a scaffolded version of our memo application we need to create two very basic files: a controller and a model.
Create a file named note.php (again, the name matters — notice how the file and the class defined here are the singular note of the database table notes) and save it in your /app/models/ directory. You need only include the following lines in it:
<?php class Note extends AppModel { var $name = 'Note'; } ?>
Similarly, create a notes_controller.php file containing the code below, and place it in /app/controllers/.
<?php class NotesController extends AppController { var $name = 'Notes'; var $scaffold; } ?>
The $scaffold variable will trigger CakePHP's default scaffolding behavior: a fully-functional Notes section will be created, and will be accessible at http://localhost/notes/.
That's all there is to it. You are now able to create, update, delete and display your notes with (literally) five lines of PHP code!
After playing with your new application for a while — feel free to create and delete a few notes — you'll start to notice its obvious limitations:
the layout is very plain, and apparently is not customizable
notes are deleted without confirmation
there's no validation for any data input by users
We'll now remove our scaffolding and start to develop something that's slightly more advanced. If you paid attention to the previous example, you will notice that no view files were created. That's because Cake uses predefined templates for scaffolding; in reality, you'll need a view for almost every action listed in your controller.
Furthermore, our controller had no actions, and that is also part of the scaffold magic. A hint for the action names could be seen in the scaffolded application's URLs as we added and removed notes, namely:
In other words, all our URLs match a common pattern: they're all written in the form /<controller>/<action>/<first_parameter>/. So we need to create at least three views for the CRUD operations — we'll name them add.thtml, edit.thtml and view.thtml — as well as a default view (index.thtml) to list and manage all of the notes. The "t" in these thtml files indicates that these files are Cake templates. And what about delete.thtml? This file does not need to be created; we'll see why shortly.
Before proceeding, remove this line from your NotesController class:
var $scaffold;
The first view we should create is a list of all the notes stored in the database, which will be the default page that displays when we access http://localhost/notes/. Create a new subdirectory named notes in your /app/views/ directory, then create a new file named index.thtml inside that. This file should contain the following code:
<h1>My Notes</h1> <table> <tr> <th>Id</th> <th>Title</th> <th>Created</th> </tr> <?php foreach ($notes as $note): ?> <tr> <td><?php echo $note['Note']['id']; ?></td> <td> <a href="/notes/view/<?php echo $note['Note']['id']?>"> <?php echo $note['Note']['title']?> </a> </td> <td><?php echo $note['Note']['created']; ?></td> </tr> <?php endforeach; ?> </table>
Note that our template code is not a complete HTML document — things like the doctype and header information for all files is also provided by the framework, and the default can of course be overridden later.
This should display a list of all the stored notes, but if you try accessing http://localhost/notes/ right now, you'll get an error saying that the action index is not defined in your controller.
The code for this action needs to be created in your controller. It simply needs to retrieve all records from your notes database table and store them in an array. Cake achieves this task in one line of code:
function index() { $this->set('notes', $this->Note->findAll()); }
The method set is defined in Cake's Controller class, and is also inherited by AppController, NotesController and any other controller in your application. The purpose of set is to create a variable ($notes) that will be available in your default view (index.thtml), and its syntax is $this->set(string $variable_name, mixed $value).
The value of the $notes variable is a multi-dimensional array returned by $this->Note->findAll(). findAll is a method defined in Cake's Model class, which fetches all records in the database table associated with the model. In this example, we'll access our Note model and call the method from our controller.
Assuming that your notes table has some records, the output of findAll will be something like this:
// print_r($notes) output: Array ( [0] => Array ( [Note] => Array ( [id] => 1 [title] => First note's title [body] => Some text. [created] => 2006-04-20 14:21:42 [modified] => ) ) [1] => Array ( [Note] => Array ( [id] => 2 [title] => Title... [body] => body text [created] => 2006-04-20 17:22:23 [modified] => ) ) )
As I mentioned before, this output is accomplished with only one line of code. CakePHP dramatically reduces the amount of repetitive and boring code required in your apps, thanks to its efficient built-in classes and intuitive conventions.
We proceed similarly to view a single note. First, we need a view.thtml view file in our /app/views/notes/ directory:
<h1><?php echo $data['Note']['title']?></h1> <p><small> Created: <?php echo $data['Note']['created']?> </small></p> <p><?php echo $data['Note']['body']?></p>
Then, we add the corresponding view action to our controller:
function view($id) { $this->Note->id = $id; $this->set('data', $this->Note->read()); }
This method takes one parameter: the ID of the note we want to view ($id). In order to retrieve a particular note, we have to set the $id variable of our Note model to the $id parameter we passed to the method. Then we create a $data variable, which is available in our view via the set method. It contains an array returned by $this->Note->read(). read fetches only one row from our notes table, which corresponds to a particular $id.
Next, we'll create a view to add a new note. All we need is a file named add.thtml in the /app/views/notes/ directory:
<h1>Add Note</h1> <form action="<?php echo $html->url("/notes/add"); ?>" method="post">
<p> Title: <?php echo $html->input('Note/title', array('size' => '40'))?> </p> <p> Body: <?php echo $html->textarea('Note/body') ?> </p> <p> <?php echo $html->submit('Save') ?> </p> </form>
This code creates a basic form that allows users to enter a title and text for a note, and to save it. This time, I decided to use some convenience code to create the two input tags via the so-called HTML Helper. Helpers will be discussed in detail in the next section of this article, but to be brief, they are classes that are accessible from views, and they contain useful methods for formatting text, creating tags, adding Javascript [19] or AJAX [20] code, and so on. The HTML Helper is available by default in all views, and is used to create (X)HTML tags. I used it in this view to create an input tag, a textarea and a submit button. The syntax is relatively straightforward, but it's important to note that in order to map the input fields to our table columns easily, and thus automate the insertion process, the names of the input fields (usually the first parameter of each method of the HTML Helper) must be in the form <model_name>/<table_field>.
The add method for the Notes Controller can be something like this:
function add() { if (!empty($this->data['Note'])) { if($this->Note->save($this->data['Note'])) { $this->flash [21]('Your note has been updated.','/notes/'); } } }
First of all we check whether or not the $this->data variable — a sort of "optimized" version of the $_POST array — is empty. If it contains something, that data is automatically saved in your notes table through the $this->Note->save() method call.
The flash method that's called afterwards will be familiar to anyone who has dabbled in Rails: it's used to keep small amounts of data in between requests, such as error messages or warnings; in this case it displays a temporary message for a few seconds, then redirects the user to http://localhost/notes/.
![]() |
The created and modified fields of our notes table are automatically populated with relevant data whenever a note is added or modified via the save method, so there's no need to keep track of those actions manually. Pretty useful, hey? |
At this point you should notice that something is wrong. The add.thtml view and the add action described above are potentially very, very dangerous in their simplicity: there is no data validation whatsoever, so, at the moment, any kind of data entered by our users will be stored in our database without being filtered or checked. Cake has some built-in validation and input sanitizing mechanisms (which we'll examine briefly in the next section), but we'll keep things simple for now, as this is just a very elementary example to introduce CakePHP's basic features.
Editing a note is similar to adding a new one, the difference being that the edit form's values must already contain data.
/app/views/notes/edit.thtml:
<h1>Edit Note</h1> <form action="<?php echo $html->url('/notes/edit')?>" method="post"> <?php echo $html->hidden('Note/id'); ?> <p> Title: <?php echo $html->input('Note/title', array('size' => '40'))?> </p> <p> Body: <?php echo $html->textarea('Note/body') ?> </p> <p> <?php echo $html->submit('Save') ?> </p> </form>
/app/controllers/notes_controller.php:
function edit($id = null) { if (empty($this->data['Note'])) { $this->Note->id = $id; $this->data = $this->Note->read(); } else { if($this->Note->save($this->data['Note'])) { $this->flash('Your note has been updated.','/notes/'); } } }
In this case, if no data is submitted, the values from the record we want to edit are retrieved and displayed in the view. Otherwise, if data is submitted, the record is updated via the save method as usual. Again, there are some obvious limitations to this simple function:
We do not validate, filter or check the $id parameter (in reality, we should make sure that the $id is numeric and that it actually exists).
Submitted data is not validated or filtered.
No error handling occurs — if something goes wrong, the user will never receive a warning message.
Finally, in order to delete a note, all we need to do is create a delete action in our NotesController; no view file is necessary, since users will be redirected to the index page, where a message will be displayed.
/app/controllers/notes_controller.php:
function delete($id) { if ($this->Note->del($id)) { $this->flash('The note with id: '.$id.' has been deleted.', '/notes'); } }
After defining all of our CRUD operations, we can make the interface easier to use by adding some convenient links for adding, editing and deleting notes. We can also rewrite our index.thtml view using the HTML Helper:
<h1>My Notes</h1> <p> <?php echo $html->link('Add Note', '/notes/add') ?> </p> <table> <tr> <th>Id</th> <th>Title</th> <th>Created</th> </tr> <?php foreach ($notes as $note): ?> <tr> <td><?php echo $note['Note']['id']; ?></td> <td> <?php echo $html->link($note['Note']['title'], "/notes/view/{$note['Note']['id']}")?> [<?php echo $html->link('Edit', "/notes/edit/{$note['Note']['id']}")?>, <?php echo $html->link('Delete', "/notes/delete/{$note['Note']['id']}", null, 'Are you sure?')?>] </td>
<td><?php echo $note['Note']['created']; ?></td> </tr> <?php endforeach; ?> </table>
In this example, I used the $html->link() method call, which is able to easily create "Cake-friendly" links. It can take up to six parameters:
the text of the link
the internal URL
an array of HTML attributes (if any)
text for a Javascript confirmation message
whether we want to convert special characters in the title to HTML entities
whether this method should either return or output a value link($title, $url=null, $htmlAttributes=null, $confirmMessage=false, $escapeTitle=true, $return=false)
The complete controller should look like this:
<?php class NotesController extends AppController { var $name = 'Notes'; function index() { $this->set('notes', $this->Note->findAll()); } function view($id) { $this->Note->id = $id; $this->set('data', $this->Note->read()); } function add() { if (!empty($this->data['Note'])) { if($this->Note->save($this->data['Note'])) { $this->flash('Your note has been updated.','/notes/'); } } }
function edit($id = null) { if (empty($this->data['Note'])) { $this->Note->id = $id; $this->data = $this->Note->read(); } else { if($this->Note->save($this->data['Note'])) { $this->flash('Your note has been updated.','/notes/'); } } }
function delete($id) { if ($this->Note->del($id)) { $this->flash('The note with id: '.$id.' has been deleted.', '/notes'); } } } ?>
Not too difficult, is it? Granted, if you're not accustomed to the MVC pattern, this might all seem a bit strange, but our PHP code definitely looks much more organized and it's much easier to maintain than most unstructured PHP architectures.
One thing to keep in mind is that all those little conventions used in Cake actually matter: for example, the name of the controller must be plural and the model must be singular, while database tables should be plural (CakePHP's Inflector class does the rest), views must be placed in a folder named after the controller, and so on. Yes, you can get around some of these conventions, but it is precisely these details that make Cake virtually self-configuring: it's a case of convention over configuration, exactly like Rails. CakePHP may not be not the best solution for everybody, but it's certainly a simple and intuitive way to solve many of the problems associated with web development.
At this point, you probably have a lot of questions. For example, I wrote that CakePHP has a native validation mechanism and it can sanitize data. What does that mean? Why didn't we modify our model class? We'll answer these and other questions in the next section.
CakePHP comes with some very handy helpers that can really make your life easier when it comes to creating views:
HTML — allows quick creation of HTML tags, including links and input fields
JavaScript — offers an easy way to manage JavaScript code
Number — a set of useful methods to format numeric data
Time — functions to format time strings and timestamps
Text — auto-link URLs, truncate strings, create excerpts, highlight, strip links and more
AJAX — a truly amazing AJAX helper, to be used in conjunction with the popular Prototype and script.aculo.us libraries; this helper can really speed up the creation of AJAX interfaces
More information about helpers is available in the CakePHP manual [27].
By default, a NotesController will try to locate and load a Note model class. If your controller needs to access more than its default model, you can define additional models by setting the $uses array, like this:
var $uses = array(Note, AnotherModel, YetAnotherModel);
In some cases, two or more tables might be closely related and would therefore be used with JOIN statements: your notes may have been submitted by different people listed in an authors table, for example. In these cases, CakePHP's Associations can be used to define complex table relationships directly in your Model class. More information is available in these manual pages.
Good question. You will almost always have to create some complex logic for an application, and you usually want to re-use part of that logic. The most common way to include an application-wide function or variable so that it's available in every controller is to define it in your AppController file. This file basically consists of an empty class that extends Cake's internal Controller class, and is located in the /cake/ directory. You can move it to your /app/ directory and create methods that will be available in all of your custom controllers that extend AppController. Even if you're not planning to use an AppController at first, it's often wise to create custom controllers which extend AppController rather than the Controller class.
An easy way to create custom classes handling a specific task is to create a component. Components can be loaded automatically in controllers (and only inside controllers) by adding a variable named $components:
var $components = array('Session', 'MyCustomComponent');
CakePHP comes with some default components such as Session, which offers convenient ways to organize session data, or RequestHandler, which can be used to determine more information about HTTP requests. These are documented in the CakePHP manual:
Session component manual pages [25]
Request Handler component manual pages [26]
documented on: 2007.01.26