I'm currently working with a CodeIgniter3 application, and have built the basis for a standard blog-like system.
The structure is standard CI - User requests a page, page loads controller method, controller method calls any relevant DB functions from its model, and then a view is loaded.
However, I'm looking to also make this view accessible via an API. So, instead of the $data array being filled with information to populate into HTML, I would instead pass it to a different view which would output a JSON result.
I imagine writing two different controllers would be a bad step to take - is there any routing I can do, or any standard practice to allow for the controllers to recognise an api endpoint has been hit (such as the directory 'api' being in the access path), and then load up a different view based on this?
For an API, I wouldn't use a view. Views are traditionally for HTML. I'd suggest instead of passing $data to a view, simply echo it out at the end of the controller like so.
echo json_encode($data);
I've created this wrapper to make extending how I return data flexible. Here's a basic example.
function api_response($data = array()) {
$data['error'] = false;
function filter(&$value) {
if (is_string($value)) {
$value = htmlspecialchars($value, ENT_QUOTES, 'UTF-8');
$value = nl2br($value);
}
}
array_walk_recursive($data, "filter");
return json_encode($data);
}
And I use something like this for API errors
function api_error_response($error_code, $error_message) {
log_message('error', $error_code . ' - ' . $error_message);
$data['error'] = true;
$data['error_code'] = $error_code;
$data['error_message'] = $error_message;
return json_encode($data);
}
And then I call it like so at the end of a controller.
echo api_response($data);
Additionally, to be able to use the same controller methods for the API as you do for the web GUI, only duplicate the routes, and in the controller methods use something like this.
// Return here for API
if (strpos($_SERVER['REQUEST_URI'], '/api/') !== false) {
// Unset any data you don't want through the API
unset($data['user']);
echo api_response($data);
return false;
}
// Else, load views here