Since the last time I wrote about blog-tools
the crate has had quite a bit of development, so I thought now would be a good time to shout out about it.
First, the create now comes with three working examples of a blog you can run right away. Simply clone the repo and run
cargo run --example high
cargo run --example medium
cargo run --example low
What do the high
, medium
and low
refer to? That's the bulk of the new features! blog-tools
now has a bunch of features to help with managing RAM.
high
refers to high RAM usage - the entire blog is stored in memory the whole time. medium
refers to medium RAM usage - most of the metadata is stored in memory but the blog posts themselves stay on disc till needed. low
refers to (you guessed it) low RAM usage - nothing is stored in memory at all. This would also be suitable in a serverless application.
high
Let's look at the high
module in more detail. Before version 0.1.0, this was the only mode supported. The main crux remains the same as before - place the blog, now the HighBlog
struct, in a lazy_static
pub static BLOG_ROOT: &str = "examples/blog";
lazy_static! {
pub static ref STATIC_BLOG_ENTRIES: HighBlog =
get_high_blog(PathBuf::from(BLOG_ROOT), None, None).unwrap();
}
You can then use this in just the same was as before, for example, in Rocket we could do something like
#[get("/blog")]
fn blog_index() -> Option<Template> {
let mut context = rocket_dyn_templates::tera::Context::new();
context.insert("blog", get_blog_context());
Some(Template::render("blog_index", context.into_json()))
}
And then the template we can make cards for the blog previews like so
<div class="blog-container">
{% for entry in blog.entries %}
<div class="blog-card">
<a href="/blog/{{ entry.slug }}" class="no-underline black-links">
<div class="card-body">
<h4>
<span>{{ entry.title }}</span>
</h4>
<h5>
<span>{{ entry.date }}</span>
</h5>
<div>
{{entry.preview | safe}}
</div>
</div>
</a>
<p><code>blog-tools</code> also supports tags - try clicking one!</p>
<div class="blog-tag-container">
{% for tag in entry.tags %}
<a href="/blog/tag/{{ tag }}" class="blog-tag">#{{ tag }}</a>
{% endfor %}
</div>
</div>
{% endfor %}
</div>
I won't cover every single way you can use it here as that's detailed in the example code!
medium
With medium
we still place the MediumBlog
in a lazy static like so
pub static BLOG_ROOT: &str = "examples/blog";
lazy_static! {
pub static ref STATIC_BLOG_ENTRIES: MediumBlog =
get_medium_blog(PathBuf::from(BLOG_ROOT), None, None).unwrap();
}
The code for the preview page looks just the same!
#[get("/blog")]
fn blog_index() -> Option<Template> {
let mut context = rocket_dyn_templates::tera::Context::new();
context.insert("blog", get_blog_context());
Some(Template::render("blog_index", context.into_json()))
}
As does the template code, so I won't repeat it here. The difference comes when we want to return a specific blog post.
#[get("/blog/<date>/<slug>", rank = 2)]
fn blog_article(date: String, slug: String) -> Option<Template> {
let mut context = rocket_dyn_templates::tera::Context::new();
let all_blogs = get_blog_context();
let complete_slug = format!("{}/{}", date, slug);
let this_blog = match all_blogs.hash.get(&complete_slug) {
Some(x) => x,
None => return None,
};
context.insert(
"blog",
&this_blog
.render(PathBuf::from_str(BLOG_ROOT).unwrap())
.unwrap(),
);
Some(Template::render("blog", context.into_json()))
}
Since we don't have the actual blog entry in memory, we need to call this render
method! This actually produces the HTML from the markdown on the disc then and there
low
low
works quite differently. Since we don't want to keep anything in memory here, there is no lazy static. Instead, we have to read everything from disc every time a request comes in. For instance, our main blog preview is
#[get("/blog")]
fn blog_index() -> Option<Template> {
// I only use this dummy struct to keep consistency with the other two blog modes
#[derive(Serialize, Deserialize)]
struct Blogs {
entries: Vec<PreviewBlogEntry>,
tags: Vec<String>,
}
let mut context = rocket_dyn_templates::tera::Context::new();
let preview = preview_blogs(PathBuf::from_str(BLOG_ROOT).unwrap(), 2, None).unwrap();
let tags = get_blog_tag_list(PathBuf::from_str(BLOG_ROOT).unwrap()).unwrap();
context.insert(
"blog",
&Blogs {
entries: preview,
tags,
},
);
Some(Template::render("blog_index", context.into_json()))
}
Notice in the line
let preview = preview_blogs(PathBuf::from_str(BLOG_ROOT).unwrap(), 2, None).unwrap();
The second argument, the 2 in this case, is specifying the number of blog posts we want it to preview. Since low
is intended for instances where the whole blog can't fit into memory at once, you almost certainly can't preview the entire blog. So, this will only bother to read the most recent N blogs!
When we want to get a specific blog
#[get("/blog/<date>/<slug>", rank = 2)]
fn blog_article(date: String, slug: String) -> Option<Template> {
let mut context = rocket_dyn_templates::tera::Context::new();
let blog_post =
render_blog_post(PathBuf::from_str(BLOG_ROOT).unwrap(), date, slug, None).unwrap();
context.insert("blog", &blog_post);
Some(Template::render("blog", context.into_json()))
}
We again have to read it from the disc.
Conclusion
I hope you find this crate helpful! You can find the repo here, the crates.io entry at here and the documentation here.
As of now, blog-tools
is still very much a personal project I built for my own website. If you use it, I'd love to hear your feedback!! You can open issues on the GitHub, or email me