This application demonstrates use of the SimpleFM\ZF2 package as the Model layer of a Zend Framework 2 MVC application, using the FileMaker Server 12 FMServer_Sample database as a data provider.
This version adds a demonstration of the SimpleFM implementation of Zend\Authentication which shows best practices for handling user authentication against a FileMaker system. See Authentication Module section below for details.
SimpleFM, the examples and this documentation are tailored for PHP 5.3 and FileMaker Sever 12
- PHP 5.3.3+
- FileMaker Server 12+
With minimum effort, you might be able to get them to work with PHP 5.0 (requires SimpleXML) and any version of FileMaker server that uses fmresultset.xml grammar, however, this is not recommended.
SimpleFM is free for commercial and non-commercial use, licensed under the business-friendly standard MIT license.
- Setup a project directory in your development environment with /public defined as a vhost.
- Fork SimpleFM_FMServer_Sample and clone it into your project directory.
- Upload
FMServer_Sample_Web.fmp12
to a FileMaker Server 12 host. See /documentation/README.md for details. - Copy
/config/autoload/local.php.dist
as/config/autoload/local.php
and edit hostname as needed if not using localhost. - In your project directory, run
sudo ./composer.phar install
on the command line. See getcomposer.org for details
After completing these steps successfully, when you visit your vhost in a browser, you should see a success message if everying installed correctly.
When designing a FileMaker Web API, it is recommended that you use a dedicated FileMaker Web interface file that contains references to your main solution, but contains no data of its own. This allows you to clearly separate the Web Layouts and Table Occurrences (TOs) from the FileMaker Pro Layouts and TOs. If you have FileMaker scripts in the main solution that need to be called from your Web application, make wrapper scripts in the Web interface file to maintain a crisp separation of concerns. Maintaining a strict decoupling of your Web API may take a little bit of extra effort in some cases, but is an important best practice.
This example includes a demonstration of using the dispatch event to check the session for a valid user identity before a controller can render anything for the user, without having to to make the controllers aware of the Authentication module. It also contains an example of Zend\Form and the SimpleFM Identity class to make authentication very easy to bolt on to any ZF2 project.
ZendFramework 2 provides a robust MVC implementation, as do many PHP frameworks. Use of MVC on PHP web apps is well documented many places on the Web, although much of the attention seems to go to the View and the Controller layers. In part this may be because these two layers are more similar from one appliction to the next, and therefore are more suited to encapsulation in frameworks. MVC frameworks have to leave more of the Model layer up to the developer, as so much of it depends on both the business domain of the application (which will vary widely), and on the persistance technology employed. There are a number of good object oriented PHP model frameworks, such as Doctrine ORM, which incorporate well with MVC frameworks, but ORMs generally rely on a SQL-based DBAL (Database Abstraction Layer), so the FileMaker Server XML API cannot be adapted to work with them.
Even with FileMaker as a given, there is still wide variation in the business domain between projects. FileMaker puts some bounds on this variation by virtue of the fact that a FileMaker solution has a Relationship Graph. The Relationship Graph requires the developer/architect to think in terms of object associations, and provides some strong clues as to what the domain model should be on the Web side of the application.
So ZendFramework 2 has the View and Controller well handled; team it up with SimpleFM to define the domain Model, and you've got three solid legs for a FileMaker-backed MVC Web application.
MVC is important, but it is ZF2's powerful new DependencyInjection (Di) container that makes it an important evolution of PHP frameworks. Di makes the framework inherently modular and extensible. Using Di, it is simple to extend the core MVC framework and to leverage third-party libraries. Whereas Soliant\SimpleFM\Adapter is a stand-alone component with no inherent external dependencies, the Soliant\SimpleFM\ZF2 package combines ZF2 with SimpleFM\Adapter and mixes in some Doctrine where common OOP object modeling problems have already been solved.
In a Web application that will use the FileMaker XML API, the architect needs to define Web domain objects that represent the FileMaker domain model as defined in the FileMaker Relationship Graph. In Doctrine ORM, domain objects are referred to as Entities, and we will borrow that term here. Defining an Entity for each table allows you to cast your FileMaker data as objects with specific associations to each other that follow the domain associations defined in the FileMaker Relationship Graph. If you don't develop a domain model on the PHP side, you will be forced to juggle untyped arrays of strings that have no inherent associations with each other. This may be fine for very simple Web applications, but the initial investment in setting up your model is fairly low, so even for small projects it is probably worth while. Managing the domain model on the PHP side also allows you to cast your Entity properties. For example, you can cast object properties as dates, numbers, collections, etc.
Generally you will define one Entity per FileMaker table, but the FileMaker XML API does not give us direct access to tables. Instead we use Layouts in a manner roughly analogous to SQL views and, properly used, they can be used as Gateways to the domain model in FileMaker. For example, consider Project and Task from FMServer_Sample:
class Project
{
// Fields
protected $name;
protected $description;
// Associations
protected $tasks;
/**
* getters and setters
*/
}
class Task
{
// Fields
protected $name;
protected $description;
protected $status;
// Associations
protected $project;
/**
* getters and setters
*/
}
Table Data Gateway: An object that acts as a Gateway to a database table. One instance handles all the rows in the table. (Fowler, PoEAA, TableDataGateway).
Designate a FileMaker Layout for each Entity in your PHP domain model. A Table Data Gateway class (hereafter just Gateway) links an Entity to the Layout(s) that provide access to the Table. Additional Layouts may be defined for an entity (see next section), but a minimum of one is required by the AbstractEntity::getDefaultWriteLayoutName()
abstract static method.
Example: given a Person Table in FileMaker, define the default layout for the PersonGateway:
public static function getDefaultWriteLayoutName()
{
return 'Person';
}
In the Project example shown here, additional library dependencies are assumed to be defined in AbstractGateway
.1 These dependencies and several helper methods are omitted from the example for clarity.
AbstractGateway
provides all the basic database interaction methods shown here. They are shown as methods of Application\Gateway\Project
to illustrate the main point of a Gateway. When you implement the domain model for a Gateway, use the methods provided by AbstractGateway
as-is or override any of them with your own custom logic, and create any additional custom methods needed to support your domain model. The point is to encapsulate the inner workings of the FileMaker API in your Gateway classes, and let your ZF2 application focus only on the OO methods it needs to work with Entities. There should be no direct use of FileMaker API commands outside of the Gateway classes.
<?php
namespace Application\Gateway;
use Soliant\SimpleFM\ZF2\AbstractGateway;
use Application\Entity\Project;
class Project extends AbstractGateway
{
public function resolveEntity(AbstractEntity $entity, $entityLayout=NULL)
{
if (!empty($entityLayout)){
$this->setEntityLayout($entityLayout);
}
return $this->find($entity->getRecid());
}
public function find($recid)
{
// Call FileMaker API via SimpleFMAdapter and unserialize as Project entity
}
public function findAll(array $sort = array(), $max = NULL, $skip = NULL)
{
// Call FileMaker API via SimpleFMAdapter and unserialize as Project ArrayCollection
}
public function findBy(array $search, array $sort = array(), $max = NULL, $skip = NULL)
{
// Call FileMaker API via SimpleFMAdapter and unserialize as Project ArrayCollection
}
public function create(Project $entity)
{
// Call FileMaker API via SimpleFMAdapter and unserialize as Project entity
}
public function edit(Project $entity)
{
// Call FileMaker API via SimpleFMAdapter and unserialize as Project entity
}
public function delete(Project $entity)
{
// Call FileMaker API via SimpleFMAdapter and return TRUE on success
}
}
Data from all the fields included on a Filemaker Layout are always returned when the Layout is called by the API. There is no programatic way to modify which columns are returned, short of using a different Layout. You don't want to be forced to eagerly load the whole object in all circumstances. It can be useful to maintain one or more alternate layouts that are sub or super sets of the default fields that comprise an Entity class. For example you might define an EntityList Layout which contains the minimum properties required to identify an Entity, e.g. name, and primary key, and can be used when lazy loading is preferable. As the EntityList is an incomplete representation of the Entity, any instance retrieved with it must be treated as read-only, and the AbstractGateway provides a method for resolving the Entity from an incomplete Layout. For instance, if you specify the EntityList Layout to generate a list view, you can pass a sparse Entity instance from that collection to the resolveEntity() method on the EntityGateway, and it will make a request for that entity and return it with a serializable field set:
// retrieve the gateway and specify a sparse layout
$gatewayProject = $this->getServiceLocator()->get('gateway_project');
$gatewayProject->setEntityLayout('ProjectList');
// execute the request
$projects = $gatewayProject->findAll();
// pick one of the result records and resolve just that record with a second request on a full layout
$sparseProject = $projects[0];
$fullProject = $gatewayProject->resolveEntity($sparseProject, 'Project');
Footnotes
Footnotes
-
The constructor requires an instance of
Zend\ServiceManager
,Soliant\SimpleFM\ZF2\Entity\AbstractEntity
,Soliant\SimpleFM\Adapter
, injected via a factory closure in the service_manager section of module.config.php.AbstractGateway
also usesDoctrine\Common\Collections\ArrayCollection
. ↩