Skip to content

Latest commit

 

History

History
246 lines (199 loc) · 10.5 KB

README.md

File metadata and controls

246 lines (199 loc) · 10.5 KB

HTML-go

HTML-go is a databaseless, flat-file blogging platform, which is very simple, fast and flexible.

Scrutinizer Code Quality Build Status Code Intelligence Status Maintainability Quality Gate Status

Features

  • Simple, fast and flexible
  • Supports PHP 8+
  • Multiple template systems supported: Twig, Smarty and PHP
  • Categorization and multiple tags
  • Content format in JSON
  • GitHub flavoured Markdown
  • Content summary through front matter or manual splitting
  • Multiple menus defined through front matter
  • Nested static pages
  • i18n support
  • Admin console (work-in-progress)

Landing Pages

HTML-go has four editable landing pages for blog, category, tag and the index or home page which at the root of all the other static pages. All of these pages are listed in the slugIndex and the pageIndex and are considered to be pages by the html-go.

Home Page (main index)

The data file is located at content/common/pages/index.md and is listed in two indexes: slugIndex and pageIndex under the key /.

Category Index Page

The data file is located at content/common/landing/category/index.md and is listed in two indexes: slugIndex and pageIndex under the key category.

Blog Index Page

The data file is located at content/common/landing/blog/index.md and is lised in one index: slugIndex under the key blog. Generally, this index page is use if the 'blog' link is enabled which should link to this page.

Tag Index Page

This data file is located at content/common/landing/tags/index.md and is listed in one index: slugIndex and pageIndex under the key tag. This is the only data file associated with tags. A tag does not have an associated file on the filesystem.

Special Pages

HTML-go has some special pages:

404 Not Found

This page must be named not-found.json and located as shown below:

content
├───common
│   ├───page
│   │   ├───not-found.json

Internally, a resouce which cannot be located is mapped to the key not-found.

Indexing

The indexing system is at the core of HTML-go. All content is listed in one or more indexes. The main index is called the slugIndex and list all posts, categories, pages and tags.

There are some special composite indexes for category to posts and tag to posts.

Routing

There is no complex router for html-go. Rather, HTML-go uses a indexing system whereby all the content is indexed with its unique URI used as the index key. Apart from a few special cases such as landing pages, the requested URI is passed to the indexing system to check if it exists, if it does it is loaded and rendered. Otherwise the not found page is rendered.

Content

Content files are in JSON format as JSON is handled natively by PHP and conversion between JSON object, stdClass and an array is also handled natively. The minimum required for a valid content file is:

{
    "title": "some title",
    "description": "some description",
    "body": "The content of this article."
}

HTML-go automatically generates a summary of the content.

Front Matter Content Summary

If you require the summary to be something different to the opening text of the article, you can add the following to the front matter:

{
    "title": "some title",
    "description": "some description",
    "summary": "Now for something completely different."
    "body": "The content of this article."
}

Manual Content Summary

If the summary variable is not defined in the front matter, HTML-go will search the body for the divider marker <!--more--> and split the text. For example:

{
    "title": "some title",
    "description": "some description",
    "body": "The content<!--more--> of this article."
}

The above will give a summary of "The content" and the body "The content of the article."

Menus

Menus entries are valid for pages only. A single content page can be listed in as many menus as required. Defined menus are available on the content.menus.[menu_name] object within the template context.

For example, below is a sample home page with the page listed in two menus: main and footer. The name for the menu link is Home in both menus and the position (weight) is the first entry. The actual link will be the same for both menus and is defined by the system, in this case /

{
    "title" : "Our Website",
    "description" : "Welcome to our website",
    "menus": {
        "main": {
            "name": "Home",
            "weight": 1
        },
        "footer": {
            "name": "Home",
            "weight": 1
        }
    },
    "body" : "Welcome to our new website."
}

The above menus can be accessed by the following Twig code:

{{ content.menus.main }}

and

{{ content.menus.footer }}

Twig Code Sample

{% if content.menus.main is defined %}
{% for main in content.menus.main %}
        <a href="{{ content.site.url }}{%if main.key starts with '/'%}{{ main.key }}{% else %}/{{ main.key }}{% endif %}">{{ main.name }}</a>
{% endfor %}
{% endif %}

Content is identified by its section. There are currently four sections page for static pages, tag for tags, category for categories and post for posts. All content belongs to one of these sections and this can be used within templates to identify a particular piece of content.

{% if content.section equals 'post' %}
...
{% endif %}

Templating

Context Variables

All variables are accessed via the content object.

Twig Smarty PHP Config Comments
{{ content.site.language }} ? ? site.language Default: "en". Also used for i18n.
{{ content.site.name }} ? ? site.name Default: "HTML-go". Use for page header.
{{ content.site.title }} ? ? site.title Default: " | HTML-go". Use for browser title.
{{ content.site.url }} ? ? site.url Must be configured manually.
{{ content.site.description }} ? ? site.description Use if {{ content.description }} is empty.
{{ content.site.copyright }} ? ? site.copyright Default: "(c) Copyright, Your Name"
{{ content.title }} ? ? "title": "xxx" Content front matter.
{{ content.description }} ? ? "description": "xxx" Content front matter.
{{ content.list }} ? ? N/A An array of content objects associated with this parent content object. E.g. A list of posts.
{{ content.section }} ? ? N/A The section too which the content belongs.
{{ content.summary }} ? ? N/A A summary of the content if defined. See summary.
{{ content.timestamp }} ? ? N/A A W3C formatted timestamp from the filename.
{{ content.date }} ? ? blog.post_date_format Default "F d, Y"

i18n

The i18n feature is accessed via the i18n object. This object has one method which is used to look up the appropriate text associated with the given key. For example:

{{ i18n.getText('widget.category_list.title') }}

Technical Data

Content Object

The content object is a stdClass and has the following public properties:

JSON and Markdown

PHP's decode_json() function does not like newline characters (\n) and will return an error if it encounters this in the data being decoded. Therefore, HTML-go uses a marker (acceptable to JSON) within the markdown text to indicate a newline character. After decoding, this marker is replaced witin the text by a newline character.

File Naming Convention

Filenames have significance in HTML-go.

Filename

| Timestamp   | Tag list    | Slug   | Ext |
20210223123423_tagone,tagtwo_the-slug.json

Timestamp

year(4)
month(2)
day(2)
hour(2)
minute(2)
seconds(2)

Comma-separated Tag List

The list of tags for this content, separated by a comma.

Slug

The slug for this content. This must be unique within the system. Slugs are expected use the dash ( - ) as a spearator because the underscore ( _ ) is used as a separator between the timestamp, tag list and the slug; and this is recommended for good SEO.

Extension

All content files use the JSON format and must have the .json file extension.

File Location

File locations have significance in HTML-go.The content directory is expected to follow a particular layout. This layout is as follows:

content
├───common
│   ├───category
│   │   ├───index.json
│   │   ├───uncategorized.json
│   │   ├───[category name].json
│   │   ├───...
│   ├───page
│   │   ├───index.json
│   │   ├───[page name].json
│   │   ├───[dir name]
│   │   │   ├───index.json
│   │   │   └───...
│   │   └───...
│   ├───post
│   │   └───index.json
│   └───tag
│       └───index.json
└───user-data
    └───[username]
        └───post
            └───[category name]
                └───[post type]

index.json

In the representation above, index.json file are landing pages for the various sections. The index.json directly under the page section (content/common/page) is the home page for the website.

Categories

There is a default category with the title Uncategorized and the slug category/uncategorized (i.e. section/filename minus extension). The content file is located in content/common/category/uncategorized.json.

Pages

In theory, pages can have infinite depth (limited only by the host OS filesystem), in practice it might be a little slow to index deeply nested pages.