lexide/pro-forma

A library for managing and installing templated files for auto-generated code

Installs: 50

Dependents: 1

Suggesters: 0

Security: 0

Stars: 0

Watchers: 2

Forks: 0

Open Issues: 0

Type:composer-plugin

1.1.0 2024-10-01 11:17 UTC

This package is auto-updated.

Last update: 2024-10-31 11:28:51 UTC


README

This library allows other libraries to configure templates for auto-generated code, allowing for quick bootstrapping of projects, using a composer plugin.

Once the Pro Forma plugin is enabled, the templating process is seamless from an end-user perspective, so installing a library that is configured with Pro Forma templates will automatically generate those files if they don't already exist.

Project installation

To use Pro Forma in a project, enable the plugin in the composer.json file and allow PuzzleDI to use the library containing the code templates:

{
  "config": {
    "allow-plugins": {
      "lexide/pro-forma": true,
      "lexide/puzzle-di": true
    }
  },
  "extra": {
    "lexide/puzzle-di": {
      "whitelist": {
        "lexide/pro-forma": [
          "target/library"
        ]
      }
    }
  }
}

Running install or update Composer commands will invoke Pro Forma code generation. Composer may also ask if the Pro Forma plugin should be enabled, if it is not already.

It is recommended that the auto-generated files be added to version control, however this is not required.

IMPORTANT! Only whitelisted libraries will have their templates generated. This includes any other libraries that whitelisted libraries have added to the whitelist trust chain. This is intended to prevent unintentional code generation when adding a new library to a project. If Pro Forma does not generate the expected files, it is advised to check that the library is correctly whitelisted.

Template Providers

Pro Forma works by having libraries register template providers with it, using lexide/puzzle-di, then processing the template config provided by those classes to convert template files into generated code.

Pro Forma uses lexide/puzzle-di to collect a list of TemplateProviderInterface class names.

Composer configuration

Each library can register a single template provider, that implements the Lexide\ProForm\Template\TemplateProviderInterface interface, by adding the following configuration to the library's composer.json file:

{
  "extra": {
    "lexide/puzzle-di": {
      "files": {
        "lexide/pro-forma": {
          "class": "Fully\\Qualified\\Template\\Provider\\Class\\Name"
        }
      }
    }
  }
}

If this library uses dependencies that should also use Pro Forma to auto-generate code, those dependent libraries can be added to the whitelist chain for PuzzleDI in this library's composer.json file, much the same as when configuring a project:

{
  "extra": {
    "lexide/puzzle-di": {
      "whitelist": {
        "lexide/pro-forma": [
          "dependent/library"
        ]
      }
    }
  }
}

Templates

The purpose of a template provider is to return an array of instances of the Lexide\ProForm\Template\Template class. These instances need to contain all the information required to template the generated code.

In order to customise which templated files are returned, the providers are passed two config objects; a Lexide\ProForm\Template\ProviderConfig\ProjectConfig instance, which contains the project name, namespace and a list of installed packages, and a Lexide\ProForm\Template\ProviderConfig\LibraryConfig instance, which contains configuration for the current provider, taken from the project's composer.json. This allows the providers to make decisions such as adding a particular template if a package is installed or setting the value of a replacement based on project config options, for example.

The values contained in the LibraryConfig instance for a provider is defined only for its library and is configured as follows:

{
  "extra": {
    "lexide/pro-forma": {
      "config": {
        "target/library": {
          "foo": "bar"
        }
      }
    }
  }
}

In this example, the LibraryConfig instance would contain the value bar under the key foo.

$value = $libraryConfig->getValue("foo");    // value is set to 'bar'

TemplateProvider Messages

TemplateProviders have the ability to pass messages back to Pro Forma, in order to relay information to the user. This is useful when a provider has made a decision based on installed packages or config that ths user need to be informed about, such as skipping templates if a package is not installed or notifying about invalid configuration.

The TemplateProvider is responsible for implementing the getMessages() method, which should return an array of strings representing the messages that need to be displayed. If a provider doesn't supply any messages, an empty array should be returned.

Template class

The template class, Lexide\ProForma\Template\Template, has four properties:

  • name - the name of the template.
  • templatePath - the path, relative to the library root, of the template file to process.
  • outputPath - the path, relative to the project root, of the file to be generated.
  • replacements - an array of key/value pairs to substitute into the template.

These properties can be set via setters on a manually created template instance, or via the factory method: Lexide\ProForma\Template\TemplateFactory::create().

Replacements

Template files can contain placeholder values that will be substituted when the output file is generated. The substitutions array is determined by the template providers and is custom per template. Pro Forma doesn't know beforehand which placeholder values exist for a template, it will blindly attempt to replace them based on the key that the provider supplies, therefore if placeholders are used, there must always be a replacement added to the template instance for each one.

Placeholder keys are wrapped in double curly braces, {{ key }}; spaces inside the braces are optional. When defining the replacements, only use the key name, no braces are required:

$template->setReplacements(["key" => "value"]);

IMPORTANT! For consistency, replacement values must always be strings; any value that is not a string will be ignored, and a message will be outputted describing the value as invalid. This includes values that are "stringable" or that PHP would naturally convert to a string.

Regenerating files

On occasion, it may be necessary to regenerate files if a library has had an update that changes a template's behaviour.

In order to do this, temporarily add the following value to the lexide/pro-forma section in extra:

{
  "extra": {
    "lexide/pro-forma": {
      "overwrite": true
    }
  }
}

Please note that Pro Forma will continue to overwrite files while this value is set to true; it is not intended to be used permanently, to avoid unexpected changes in behaviour after a composer update or install.

Also, this setting tells Pro Forma to overwrite all files, so any customisations that aren't in version control will be removed.

Finally, Pro Forma will never delete files; if a library is uninstalled, it's auto-generated code will need deleting manually.