cspray / architectural-decision
A library to keep track of architectural decisions with PHP annotations.
Installs: 2 606
Dependents: 1
Suggesters: 0
Security: 0
Stars: 7
Watchers: 3
Forks: 1
Open Issues: 4
Requires
- php: ^8.1
Requires (Dev)
- cspray/phinal: ^2.0
- phpunit/phpunit: ^10.1
- vimeo/psalm: ^5.11
This package is auto-updated.
Last update: 2025-01-17 16:08:54 UTC
README
An Architectural Decision is a design decision that could potentially have a large impact on your codebase. Why those decisions were made, from a technical perspective and a business perspective, are important, so they should be documented properly. This library allows you to document Architectural Decision Records (ADR) as an Attribute within your codebase. Doing so provides some functionality that might be useful:
- Architectural Decisions are close to your codebase. As in, the decision is right there in the repo, and you don't have to hunt through another system to find it.
- Architectural Decisions are code in your codebase. As an Attribute you can mark the places in your code that are impacted by this decision. This makes it easier to remember for existing maintainers and easier for new developers to realize there's relevant information. If you implement the ADR following the conventions of this library PHPStorm and other IDEs will show you that decision simply by hovering over the Attribute.
- Statically analyze the impact of your decision. Over time as more of your codebase becomes annotated with Attributes you may be able to glean more information about the decision.
Installation
composer require cspray/architectural-decision
Requirements
Before using this library there are a couple assumptions that I make, especially when using the provided bin/architectural-decisions
CLI tool. If you're using Composer for installing and autoloading the assumptions are pretty safe, but they're worth pointing out.
- There's a
composer.json
file present in the root of your project. - That configuration has an
autoload
with apsr-4
orpsr-0
entry that specifies 1 or more directories in the root of your project. - Inside the directories specified in your autoloader at least 1 Attribute implementing
ArchitecturalDecisionRecord
exists. - All
ArchitecturalDecisionRecord
implementations should be able to be created with no constructor dependencies.
If these assumptions don't hold up for you the bulk of the functionality provided by this library is in the Cspray\ArchitecturalDecision\ArchitecturalDecisionAttributeGatherer
and the Cspray\ArchitecturalDecision\XmlDocumentGenerator
implementations. It would be fairly straightforward to implement this library with your own assumptions.
Usage Guide
The first thing to do is implement an Architectural Decision Record! This is handled with the Cspray\ArchitecturalDecision\ArchitecturalDecisionRecord
interface. I recommend you use the abstract Cspray\ArchitecturalDecision\DocBlockArchitecturalDecision
class. This implementation will use the DocBlock for the Attribute as the contents of the decision.
<?php declare(strict_types=1); // src/ArchitecturalDecisions/MyFirstDecision.php namespace Acme\ArchitecturalDecisions; use Cspray\ArchitecturalDecision\DecisionStatus; use Cspray\ArchitecturalDecision\DocBlockArchitecturalDecision; use Attribute; use DateTimeImmutable; /** * Explain the decision and its potential business impact. */ #[Attribute] final class MyFirstDecision extends DocBlockArchitecturalDecision { public function date() : DateTimeImmutable { return new DateTimeImmutable('2022-07-19'); } public function status() : string|DecisionStatus { return DecisionStatus::Draft; } }
Optionally, you can also annotate appropriate places in your codebase where it might make sense to do so. After that you can run a command to generate an XML document detailing all of your decisions and where they were annotated in your codebase.
./vendor/bin/architectural-decisions
If successful a file named architectural-decisions.xml
will be generated and stored in the root of your project. You can use this file to generate views into the content, assist in static analysis, or whatever else you might want to do with the ADR information and where it is used in your codebase.
Setting Custom Meta Data
There might be additional information you'd like to include with an ArchitecturalDecisionRecord that doesn't fit into the contents of the decision. Perhaps it is additional data that can be used with static analysis. Perhaps you like to include information about who authored the decision or some other meta-data. You can implement the ArchitecturalDecisionRecord::setMetaData(DOMElement $meta)
method to add whatever data you'd like to the generated XML document. Please review the DOMDocument documentation for how to appropriately add elements and attribute to the <meta>
element.
Example XML Document
This document was generated by parsing this library. We include a lightweight example for why you should use Attributes for ADR!
<?xml version="1.0" encoding="UTF-8"?> <architecturalDecisions xmlns="https://architectural-decision.cspray.io/schema/architectural-decision.xsd"> <architecturalDecision id="UsingAttributesForArchitecturalDecisions" attribute="Cspray\ArchitecturalDecision\ArchitecturalDecisionRecords\UsingAttributesForArchitecturalDecisions"> <date>2022-07-19</date> <status>Accepted</status> <contents><![CDATA[Architectural Decision Records (ADR) can be useful in determining why a piece of software is the way it is. While these type of documents can live anywhere, an Attribute in your codebase can be a good place to store this info. For more information, please check out the README in this repo or at https://github.com/cspray/architectural-decision]]></contents> <codeAnnotations> <codeAnnotation> <class>Cspray\ArchitecturalDecision\ArchitecturalDecisionRecord</class> </codeAnnotation> </codeAnnotations> <meta /> </architecturalDecision> <!-- Additional architecturalDecision would be listed here --> </architecturalDecisions>