cspray / blogisthenics
Put your blog generation on a diet with Blogisthenics!
Requires
- php: ^8.1
- adbario/php-dot-notation: ^3.1
- cspray/annotated-container: ^1.2
- laminas/laminas-escaper: ^2.10.0
- league/commonmark: ^2.3
- minicli/command-help: ^0.1.0
- minicli/minicli: ^3.2
- monolog/monolog: ^3.1
- rdlowrey/auryn: ^1.4
- voku/stringy: ^6.5
Requires (Dev)
- mikey179/vfsstream: ^1.6
- phpunit/phpunit: ^9.5
This package is auto-updated.
Last update: 2024-12-22 02:40:25 UTC
README
Are you looking for a static site generator with the following features:
- A SPA using the latest JavaScript Web Components?
- The newest CSS frameworks baked right in?
- Access to the fanciest, most modern templating engine?
- Integration with your favorite framework?
Then this project isn't for you because Blogisthenics doesn't offer any of those things! And never will! What do we give you instead?
- A way to create a boring ol' multi-page application using HTML, Markdown, and a minimal amount of CSS; where clicking on a link makes the whole page refresh. So old school!
- Powerful, no-frills templating engine that's been in use for over 20 years. PHP itself!
- Customize your dynamic content with Front Matter... not written in YAML!
- Absolutely no JavaScript or related tooling out-of-the-box. I prefer to have just 1 shitty language in my site generators, thank-you-very-much!
- Cohesive, type-safe, testable mechanisms for programmatically controlling the content of your site!
Usage Guide
Oh, shit. You're still here? In all honesty, you probably shouldn't use this software! There's just an ass-load of site generators out there and nearly all of them are going to be more supported than whatever this ball of crap turns into. The stuff I write below is mostly for my own benefit so when I come back here in 6 months I can figure out what the hell is going on.
Directory Structure
Blogisthenics follows a principle that there are reasonable defaults for your site configuration, but you can override any of the defaults to customize your installation. Your directory structure should resemble the following:
/.blogisthenics
config.json # Configure Blogisthenics, if not provided default config will be used
/content # The actual content for your site goes here
/assets # CSS, JS, images ... isn't treated specially, convention to put stuff here
/blog # Your blog articles ... could be named whatever you want
index.md # Markdown files are ok. Front-matter parsing and layouts are supported
about.html # HTML files are ok too, you won't get any parsing or layout support though
contact.html.php # Add a PHP extension to enable front-matter parsing and layout support
/data
... # Store JSON files here to access in the KeyValueStore
/layouts
main.html.php # Store PHP template files here to use as layouts
article.html.php # Layouts can be nested as deep as you want, but probably a logical limit
foo.md.php # Support Markdown templates
/src
/ContentGeneratedHandler
... # Any ContentGeneratedHandler instances
/ContentWrittenHandler
... # Any ContentWrittenHandler instances
/DataProvider
... # Any DataProvider instances
/DynamicContentProvider
... # Any DynamicContentProvider instances
/Formatter
... # Any Formatter instances
/TemplateHelperProvider
... # Any TemplateHelperProvider instances
Content Overview
Content for your site gets lumped into three categories:
- Static Assets
- Layouts
- Pages
Static Assets
Static assets are any content in your site that should not be dynamically rendered, whatever is in the file gets copied over exactly, with the same path, when the site is built. Specifically the following functionalities are not supported by static assets.
- Front Matter Parsing
- Template processing, i.e. no variables
- Multiple extension formatting support
Pages
Pages are .html
, .md
, and .php
files that act as specific content for a path that will be added to your site. Pages are expected to be only partial HTML documents and must define a layout. If a layout is not explicitly defined in the Front Matter of a page we use the default layout from the site configuration.
Layouts
Layouts are .md
and .php
files that act as the outer chrome for pages. Layouts can be inserted into other layouts. The below example demonstrates a minimal layout, typically named something like main.html.php
or default.html.php
.
<!DOCTYPE html> <html> <head> <title><?= $this->title ?? 'Blogisthenics' ?></title> </head> <body> <?= $this->yield() ?> </body> </html>
Note the call to $this->yield()
, when in a layout this is required to output the content being injected. If you attempt to call $this->yield()
from a non-layout piece of Content an exception will be thrown. Check out the "Templating" section below for more details on using Pages and Layouts.
Templating
Blogisthenics uses PHP itself as the templating engine. On top of that we add some functionality to allow the following features:
- Nesting an arbitrary level of layouts
- Auto-escaping all values, including ability to escape contextually.
- Providing access to the FrontMatter of the layout and page
- Provide read-only access to the loaded data
- Allow the creation of helper methods for outputting common pieces of content
The majority of the functionality described in this section refers to the Cspray\Blogisthenics\Context
object. This object is defined as $this
in your templates. The simplest templates might look something like the following.:
<!-- Stored in the layouts directory with the filename `main.html.php` --> <!DOCTYPE html> <html> <head> <title><?= $this->title ?? 'Blogisthenics README' ?></title> </head> <body> <?= $this->yield() ?> </body> </html>
<!-- Stored in the content directory with the filename `index.md` --> { "title": "Home Page" } # <?= $this->title ?> Yep, that's right. The Front Matter is just a JSON object. Slap a new line on the end of that bad boy, then start writing your content. The values available in the page are the values found in your Front Matter.
When built, your site would include an index.html
file that resembles the following:
<!DOCTYPE html> <html> <head> <title>Home Page</title> </head> <body> <h1>"Home Page"</h1> <p> Yep, that's right. The Front Matter is just a JSON object. Slap a new line on the end of that bad boy, then start writing your content. The values available in the page are the values found in your Front Matter. </p> </body> </html>
Template Helpers
Sometimes what you want to do might be too advanced for a static front matter or would be easier to share across many pages and layouts if it was encapsulated in a method you could invoke inside a template. Using the TemplateHelperProvider
you can add template helper methods easily and then get access to them in your templates.
To utilize template helpers you'll have to implement some PHP code. In addition to whatever your helper does you'll have to make sure it gets integrated with Blogisthenics. Fortunately, that's easy to do thanks to Annotated Container. This code should live somewhere in the src
directory of your Blogisthenics site.
<?php declare(strict_types=1); namespace Acme\BlogisthenicsDemo; use Cspray\AnnotatedContainer\Attribute\Service;use Cspray\Blogisthenics\Template\MethodDelegator;use Cspray\Blogisthenics\Template\TemplateHelperProvider; #[Service] final class MyTemplateHelperProvider implements TemplateHelperProvider { public function addTemplateHelpers(MethodDelegator $methodDelegator) : void { $methodDelegator->addMethod('myHelper', function() { return 'This is my helper content!'; }); } }
Inside your template you can now invoke $this->myHelper()
!
<!-- some file ending in .md --> <?= $this->myHelper() ?>
Auto-Escaping
Since Content can be generated at runtime with Front Matter that could include data from external sources Blogisthenics takes the stance that all of your data should be escaped properly. Utilizing the laminas/escaper project we automatically escape all values originating from the Context object. Meaning, all of your front matter, method helpers, and loaded JSON data get automatically HTML escaped at the time of output. If you don't want the value from a helper method to be escaped automatically wrap it in the value object Cspray\Blogisthenics\SafeToNotEncode
.
Future updates will add a template API for contextually aware escaping. In other words, you'll be able to implicitly and explicitly escape a piece of data with awareness of whether it is CSS data, or JS data, or HTML data.
Template Formatting
As already mentioned, we support the ability to create pages out of Markdown documents. Specifically, Blogisthenics uses GitHub Flavored Markdown provided by the league/commonmark package. After all, writing a blog in just HTML would start to get pretty old... blog articles are pretty good candidates for Markdown. Instead of baking Markdown support into Blogisthenics somewhere we expose an interface called Cspray\Blogisthenics\Formatter
that provides an opportunity for a rendered template to have some additional formatting applied. The Cspray\Blogisthenics\GitHubFlavoredMarkdownFormatter
is the implementation taking care of Markdown. You can implement your own Formatter
instance if you find Blogisthenics minimalist approach too spartan for you. We'll be taking advantage of Annotated Container to easily integrate your Formatters into Blogisthenics. This code should live somewhere in the src
directory of your Blogisthenics site.
<?php declare(strict_types=1); namespace Acme\BlogisthenicsDemo; use Cspray\AnnotatedContainer\Attribute\Service;use Cspray\Blogisthenics\Template\Formatter; #[Service] final class MyCustomFormatter implements Formatter { public function getFormatType() : string { return 'my-type'; } public function format(string $contents) : string { return 'my formatted ' . $contents; } }
Now any template you create that ends in my-type
or my-type.php
will be passed to MyCustomFormatter
before the contents are written to disk.
Dynamic Content
Sometimes it isn't possible to create all the necessary content ahead of time in static files. You may need to have access to the content of the site or some other information that requires you to provide the Content at runtime. You can take care of this in Blogisthenics by implementing the Cspray\Blogisthenics\DynamicContentProvider
interface. We'll be using Annotated Container to easily integrate your implementations. This code should live somewhere in the src
directory of your Blogisthenics site.
<?php declare(stric_types=1); use Cspray\AnnotatedContainer\Attribute\Service;use Cspray\Blogisthenics\Site;use Cspray\Blogisthenics\SiteGeneration\DynamicContentProvider; #[Service] final class MyContentProvider implements DynamicContentProvider { public function addContent(Site $site) : void { // Construct your Content and call $site->addContent($content) } }
Programmatic Data
Sometimes your blog or site might need to include some programmatically loaded data. Inside Blogisthenics this data gets
stored in the Cspray\Blogisthenics\KeyValueStore
. You can depend on this service in your own #[Service]
and a mutable
implementation will be injected. In Template contexts you can get access to an immutable store through the Context::kv()
method, $this->kv()
in your templates. You can load data into the KeyValueStore
in 1 of 2 methods; statically or
through programmatic calls.
Loading Static Files
In your data directory, by default /data
in your site's root, all JSON files will be loaded and stored in the
KeyValueStore
. All data is namespaced with the name of the file and keys in the JSON object are accessible through
dot-notation. For example, If you have the following JSON file stored in /data/foo.json
you can make the below method
calls in your template.
{
"bar": {
"baz": {
"qux": "whatever"
}
}
}
// Inside a Page or Layout <?= $this->kv()->get('foo/bar.baz.qux') ?>
Loading Dynamic Data
Maybe you don't know ahead of time what data needs to be loaded. In those situations you should implement a
Cspray\Blogisthenics\DataProvider
instance. We'll be taking advantage of Annotated Container to easily integrate this
instance. Somewhere in your /src
directory you should implement the following:
<?php declare(strict_types=1); use Cspray\AnnotatedContainer\Attribute\Service;use Cspray\Blogisthenics\SiteData\DataProvider; #[Service] class MyDataProvider implements DataProvider { public function addData(KeyValueStore $keyValue) : void { // add whatever data is appropriate } }
Just like statically loaded files you can access whatever data you load into the store through dot-notation. Unlike statically loaded files nothing is autonamespaced. If your data needs require a namespace you should make sure you do that in the keys you use to set data.
Why?!
Holy shit, cspray! Have you never heard of Jekyll, or Hugo, or any of the 134 bajillion generators listed on the JAMStack site?
Well, clearly I know about them. I included links to their sites! There's 4 primary reasons I did this.
-
I'm a masochist.
-
When I look at site generators out there I see a lot of libraries with, from my perspective, the following drawbacks:
- Aren't written in PHP
- Create sites using SPA architectures
- Require me to use Laravel or Symfony
I want to use PHP to build my site, and without requiring Laravel or Symfony to do so. I didn't find anything targeted towards the framework-less audience, so I built it myself.
-
Opportunity to dogfood Annotated Container.
-
Creating things to learn how they work or how I might accomplish it is how I learn things. It is kinda my jam.