Spalatum is a library to merge different fragment sources into a single template. With Spalatum you can get multiple external fragments and serve all together in your html page. This project is based on Tailor.
Some of Spalatum features and benefits:
- Composes pre-rendered markup on the backend. This is important for SEO and fastens the initial render.
- Enforces performance budget. This is quite challenging otherwise, because there is no single point where you can control performance.
- Cache Management - When you use
cache
attribute in<fragment>
tag, your app performace will be improved. With spalatum's cache management we give you the power to see what is cached, delete some specific endpoint or just purge all cached fragment. - Remove duplicated assets - Spalatum avoid multiple requests for an external Javascript or CSS, preserving only the first. So, if you have 2 or more fragments importing the same JS/CSS lib, we will remove all duplicated call starting from the second.
Before start, you must understand Fragment.
We understand "Fragment" as every endpoint hosted on http(s) server that provide the content you want to include in your page. If you want to use some specific js or css, you can use script
, style
, Link
or any other tags in your template page to provide this resources. Check our example app using Spalatum for the skeleton implementation.
You can represent a Fragment using the <fragment />
tag with this attributes: href, proxy, cache.
Represent the endpoint that provide the content you want to include in your page.
This endpoint must returns 200
as status code and text/html
as content-type. Otherwise it will not be rendered.
Example:
<fragment href="http://example.fragment.com/" />
Represents the main content. You can use only once primary attribute per template. If the request throws any error, will be returned an error object instead of the rendered html.
Example:
<fragment href="http://example.fragment.com/" primary />
Error object example:
{
message: 'Spalatum can't render the primary fragment ({fragment.href}), the returned statusCode was {response.status}.',
statusCode: 500
}
If your href need to be accessed via proxy, use the complete proxy url as PROTOCOL://HOST:PORT
Example:
<fragment href="http://example.fragment.com/" proxy="https://127.0.0.1:8081" />
The presence of cache attribute is optional, if you use, the fragment will be cached as you specify, if you don't, the fragment will be requested each time the page is loaded.
Spalatum represent cache lifetime as Momentjs does.
Example:
Time | Cache attribute |
---|---|
no cache | don't use the cache attribute |
10 years | 10y |
3 quarters | 3Q |
6 months | 6M |
2 weeks | 2w |
7 days | 7d |
5 hours | 5h |
12 minutes | 12m |
3 seconds | 3s |
5000 milliseconds | 5000ms |
So, if you need to cache some fragment, you can use the cache attribute to represent how many time you want to cache.
Example:
<fragment href="http://example.fragment.com/" cache="10m" />
<fragment href="http://example.fragment.com/" cache="10d" />
If you don't want to use cache, just ommit the attribute:
<fragment href="http://example.fragment.com/" />
If you want to know how this cache works "under the hood", see the cache diagram for more details.
Use the command below to add Spalatum as your project dependencie:
# You can use yarn, as well
npm install spalatum
To get started, you can create your own app using "spalatum": "^VERSION"
as dependencie in your package.json
; or clone the app skeleton using nodejs and Spalatum (RECOMENDED) that we provide.
Given you have this template:
<html>
<body>
<fragment href="http://example.fragment.com/" />
</body>
</html>
And the route http://example.fragment.com/ responds with:
<header>
<h1>This is a Fragment</h1>
</header>
Spalatum will return:
<html>
<body>
<header>
<h1>This is a Fragment</h1>
</header>
</body>
</html>
It's possible to set a proxy server as a fragment attribute, if needed:
<!-- Note that it's required to set the protocol (`http(s)`) in the proxy url attribute -->
<fragment href="http://example.fragment.com/" proxy="http://example.proxyserver.com" />
You can set how many time you want to cache some fragment attribute using proxy too:
<!-- Note that it's required to set the protocol (`http(s)`) in the proxy url attribute -->
<fragment href="http://example.fragment.com/" proxy="http://example.proxyserver.com" cache="1d" />
You can use multiple fragments together to assembly a web application:
<html>
<body>
<fragment href="http://header.fragment.com/" cache="1w" />
<div>Main Localy Rendered App</div>
<fragment href="http://banner.fragment.com/" proxy="http://proxyserver.com" />
<fragment href="http://footer.fragment.com/" cache="1w" />
</body>
</html>
This nodejs example create a Spalatum instance, setting to it a cache object, then call the render method passing to it a template string that returns a Promise instance, which will be resolved with the parsed html or reject in error case:
const spalatum = require('spalatum');
const express = require('express');
const app = express();
app.get('/', async (req, res) => {
const template = `
<html>
<body>
<fragment href="https://github.com/catho/spalatum" />
</body>
</html>
`;
const templateResult = await spalatum.render(template);
res.send(templateResult);
});
app.listen(3000);
Also, the render method accepts a custom header object, if you need to pass headers between all fragments.
...
const headers = {
Cookies: 'foo=bar',
'Content-type': 'text/html',
}
const templateResult = await spalatum.render(template, { headers });
...
When you use the fragment
tag and add a cache
attribute, internally, the Spalatum create a cache
object that store: href (as key), content, timestamp.
cacheObject = {
'[href]': {
content: '[encrypted_fragment_content]',
timestamp: '[cache_expiration_time]'
}
}
If you want, you can manage the cache inside your Spalatum instance.
There are some methods that you can use to manage Spalatum cache: getCache, clearCacheItem, clearAllCache.
Returns the cache object.
{
'http://localhost:9000/': {
content: 'aHR0cHM6Ly93d3cueW91dHViZS5jb20vd2F0Y2g/dj1kN2w5MWduV0EzSQ==',
timestamp: '2017-10-10T15:10:33-03:00'
},
'http://localhost:9001/': {
content: 'aHR0cHM6Ly93d3cueW91dHViZS5jb20vd2F0Y2g/dj1KNXdmNlVNWkVZNA==',
timestamp: '2017-10-10T15:10:33-03:00'
}
}
Remove a specific cache item by endpoint.
Given you have some cache
:
{
'http://localhost:9000/': {
...
},
'http://localhost:9001/': {
...
}
}
When you call the method Spalatum.clearCacheItem('http://localhost:9000/')
, this specified endpoint will be removed and the cache object will be:
{
'http://localhost:9001/': {
...
}
}
Remove all cache items from cacheObject.
Given you have this cache
:
{
'http://localhost:9000/': {
...
}
}
When you call the method Spalatum.clearAllCache()
, all cached endpoints will be removed and the cache object will be:
{}
- Check the issues to ensure that there is not someone already working on it
- Check our contribution guide
Clone this repository:
git clone https://github.com/catho/spalatum
Access the folder and install the project's dependencies:
cd spalatum && npm install
Before send a pull request, always runs the unit tests:
npm test
# or
npm run coverage
Thanks goes to these wonderful people (emoji key):
Allysson dos Santos 💻 📖 👀 |
Daniel Silva 💻 📖 👀 |
Luiz Kota 💻 📖 👀 |
José Luiz Coe 💻 📖 👀 |
Gabriel Daltoso 💻 📖 👀 |
Alan Oliv. 💻 👀 |
---|
This project follows the all-contributors specification. Contributions of any kind welcome!