cw/tool

There is no license information available for the latest version (v3.1.x-dev) of this package.

v3.1.x-dev 2017-03-28 15:48 UTC

README

  • Drupal 7: Build Status
  • Drupal 8: Build Status

What does it do?

CWTool has 2 main purpose. First it is a structuring framework over the Drupal API. Second it is a helper library for PHP and Drupal.

Why Drupal needs a structuring framework? Drupal (especially the 7th version) uses associated arrays everywhere, and even the more object oriented parts of it is mostly a bunch of loosely coupled weak objects. As an example nodes are plain PHP objects containing numerous associated arrays. Work with node objects are dangerous and not sustainable. Wrapping them into controllers and models however gives you well defined business layer and safety.

Another example is forms. In a old fashioned Drupal project forms are defined and altered in hooks and plain functions without much organization. Form classes and form extenders can help creating cohesive classes and reducing redundant code to the minimum.

Other than providing classes for building up a business domain CWTool intend to provide the Drupal API through adapters to increase testability of the business layer. For example entity API and variable handling is via adaptors, which means the Drupal implementation is just one version and you can override it anytime.

And possibly a +1 purpose of CWTool is to provide dependency injection to the application via a service container. Using services from DI allows the app (and developers) to switch and replace services (such as logging, variable handling, object loading, caching - to name a few) without much hassle.

The utility part of CWTool is a pure helper function library to speed up development time by reducing redundant and error prone code. There are various utilities for strings, arrays, dates, functional style code, etc.

Common use of CWTool in a generic Drupal (7) application

  • install module (with dependencies)
  • make a dedicated app module (with composer + PSR4 autoloading)
  • implement the service container hook for DI
  • use the Drush tool to generate all the node, user and taxonomy controllers and put it into src/Controller/<ENTITYTYPE>/<CONTROLLER>.php
  • create the necessary template processors and create processor classes for each
  • implement the variable collector hook and add all app variables

For utilities the best is to go through of the documentation, tests and source code to see what is available. If you don't find what you need: add it, commit it and write a test for it.

Install

  • copy the module into the modules folder
    • even better if you add it as a git submodule
    • even better through composer: composer require cw/tool
  • update composer dependencies:
    • composer update in the module folder
  • enable cw_tool module
    • drush en cw_tool

Main features

Drush commands

Entity controller class scaffolding

Creates boilerplate PHP class code for bootstrapping.

drush cwt-sc-ctrl node blog --namespace=My\\Namespace

General development guidelines

Use entities via their controllers

Entities should have their own controller, and loaded with the controller factory.

Controller should contain the minimum necessary behavior

Controllers suppose to contain data access and minimum business logic. Controller should not generate themed output. Controller should not sanitize it's content.

Controller should hold the field names (as constants), property or state constants.

For extra behavior (such as controller rendered output, forms, ets) there should be a dedicated class implementing ControllerAware or ControllerContainer.

Services over static classes

When new class needed to wrap a functionality, a new service is preferred. Usually they contain a logger at least.

Param objects over arrays

Avoid using arrays as argument. Make a parameter object instead.

Always use the structuring classes where available

  • for new forms use FormBulider
  • for template (pre)processors use AbstractThemeProcessor
  • for blocks use Block
  • for form states use FormState or NodeFormState

Avoid code in hooks

Only keep proxy calls in hooks to the appropriate classes, but no logic at all.

Documentation

Execute Doxygen generator:

doxygen Doxygen

The Doxygen binary is a requirement.