419 lines
10 KiB
Raku
Executable file
419 lines
10 KiB
Raku
Executable file
#!/usr/bin/env raku
|
|
use v6.e.PREVIEW;
|
|
|
|
use DB;
|
|
use DB::BlogMeta;
|
|
use DB::Series;
|
|
use DB::MarkdownPost;
|
|
use DB::IdrisPost;
|
|
|
|
use Pretty::Table;
|
|
|
|
my %*SUB-MAIN-OPTS =
|
|
:named-anywhere,
|
|
:bundling,
|
|
;
|
|
|
|
#| The directory this script is located in
|
|
my IO::Path:D $default-blog-dir = $*PROGRAM.parent;
|
|
|
|
#| Default database directory
|
|
my IO::Path:D $default-db-dir =
|
|
do if %*ENV<BLOG_TEST> {
|
|
$default-blog-dir.add('test-db/')
|
|
} else {
|
|
$default-blog-dir.add('db/')
|
|
};
|
|
|
|
#| The default output directory
|
|
my IO::Path:D $default-site-dir = $default-blog-dir.add('site/');
|
|
|
|
#| The default idris ipkg
|
|
my IO::Path:D $default-ipkg = $default-blog-dir.add('projects/Idris/Idris.ipkg');
|
|
|
|
#| Initalize the database
|
|
multi MAIN(
|
|
"db",
|
|
"init",
|
|
#| The path of the database file
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
#| Overwrite an already existing database file
|
|
Bool :$force
|
|
) {
|
|
die "Database folder already exists, use --force to overwrite: {$db-dir.Str}"
|
|
if $db-dir.e && !$force;
|
|
|
|
print "Blog Title: ";
|
|
my $title = get;
|
|
print "Tagline: ";
|
|
my $tagline = get;
|
|
print "Base URL: ";
|
|
my $base-url = get;
|
|
|
|
my $meta =
|
|
BlogMeta.new:
|
|
title => $title, tagline => $tagline, base-url => $base-url;
|
|
|
|
my $db = DB::PostDB.init: $meta;
|
|
|
|
$db.write: $db-dir;
|
|
}
|
|
|
|
#| Ensure that the database loads, erroring otherwise
|
|
multi MAIN(
|
|
"db",
|
|
"verify",
|
|
#| The path of the database file
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
$db.write: $db-dir;
|
|
say "Database OK";
|
|
}
|
|
|
|
#| Create a new markdown post
|
|
multi MAIN(
|
|
"post",
|
|
"new",
|
|
"markdown",
|
|
#| The path to the markdown file
|
|
IO::Path(Str) $source,
|
|
#| The path of the database file
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
#| The date/time the post should be recorded as posted at
|
|
DateTime(Str) :$posted-at = DateTime.now,
|
|
#| Should the post be hidden from the archive?
|
|
Bool :$hidden = False,
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
my $id =
|
|
$db.insert-post:
|
|
MarkdownPost.new(
|
|
source => $source.absolute.IO,
|
|
posted-at => $posted-at,
|
|
hidden => $hidden,
|
|
);
|
|
$db.write: $db-dir;
|
|
say 'Post inserted with id ', $id;
|
|
say 'Post has slugs: ', $db.posts{$id}.all-slugs;
|
|
}
|
|
|
|
#| Create a new idris post
|
|
multi MAIN(
|
|
"post",
|
|
"new",
|
|
"idris",
|
|
#| The path to the idris file
|
|
IO::Path(Str) $source,
|
|
#| The path to the ipkg file
|
|
IO::Path(Str) :$ipkg = $default-ipkg,
|
|
#| The path of the database file
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
#| The date/time the post should be recorded as posted at
|
|
DateTime(Str) :$posted-at = DateTime.now,
|
|
#| Should the post be hidden from the archive?
|
|
Bool :$hidden = False,
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
my $id =
|
|
$db.insert-post:
|
|
IdrisPost.new(
|
|
source => $source.absolute.IO,
|
|
posted-at => $posted-at,
|
|
hidden => $hidden,
|
|
ipkg => $ipkg.absolute.IO,
|
|
);
|
|
$db.write: $db-dir;
|
|
say 'Post inserted with id ', $id;
|
|
say 'Post has slugs: ', $db.posts{$id}.all-slugs;
|
|
}
|
|
|
|
#| Update the last editied time on a post
|
|
multi MAIN(
|
|
"post",
|
|
"edit",
|
|
#| The post id to touch
|
|
Int:D $id,
|
|
#| The path of the database file
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
#| The date/time the post should be recorded as laste edited at
|
|
DateTime(Str) :$edited-at = DateTime.now,
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
my $post = $db.posts{$id.Int};
|
|
$post.edited-at.push: $edited-at;
|
|
$db.write: $db-dir;
|
|
}
|
|
|
|
#| Render the blog to html
|
|
multi MAIN(
|
|
"render",
|
|
#| The path of the database directory
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
#| The path of the output directory
|
|
IO::Path(Str) :$site-dir = $default-site-dir,
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
$db.render: $site-dir;
|
|
}
|
|
|
|
#| Provide a table of posts, in newest to oldest order
|
|
multi MAIN(
|
|
"post",
|
|
"list",
|
|
#| The path of the database directory
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
#| The number of posts to show on a single page
|
|
Int :$per-page = 10;
|
|
#| The page number to show
|
|
Int :$page = 1;
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
my @pages =
|
|
$db.sorted-posts.rotor($per-page, :partial);
|
|
my @page = @pages[$page - 1].flat;
|
|
my $table = Pretty::Table.new:
|
|
title => "Posts (page $page/{@pages.elems})",
|
|
field-names => ["ID", "Title", "Type"];
|
|
for @page -> $pair {
|
|
my $id = $pair.key;
|
|
my $post = $pair.value;
|
|
# TODO: Terminal aware truncation
|
|
my $title = do if $post.title.chars > 50 {
|
|
"{$post.title.substr(0, 50)}..."
|
|
} else {
|
|
$post.title
|
|
};
|
|
my $type = do given $post {
|
|
when MarkdownPost {
|
|
"md"
|
|
}
|
|
when IdrisPost {
|
|
"idr"
|
|
}
|
|
default {
|
|
""
|
|
}
|
|
}
|
|
$table.add-row: [$id, $title, $type];
|
|
}
|
|
say $table;
|
|
}
|
|
|
|
#| Display information about a post
|
|
multi MAIN(
|
|
"post",
|
|
"info",
|
|
#| The id of the post
|
|
Int $id,
|
|
#| The path of the database directory
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
#| Display all of the information and not just the primaries
|
|
Bool :$full,
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
my $post = $db.posts{$id.Int};
|
|
given $post {
|
|
say 'Title: ', .title;
|
|
say 'Type: ', .WHAT.^name;
|
|
say 'Source: ', .source.relative;
|
|
if .hidden {
|
|
say "Hidden";
|
|
}
|
|
if .all-slugs {
|
|
if $full {
|
|
say 'Slugs: ';
|
|
for .all-slugs -> $slug {
|
|
say ' * ', $slug;
|
|
}
|
|
} else {
|
|
say 'Primary Slug: ', .all-slugs[*-1];
|
|
}
|
|
} else {
|
|
say 'Slugs: []';
|
|
}
|
|
if .tags {
|
|
say 'Tags:';
|
|
for .tags -> $tag {
|
|
say ' * ', $tag;
|
|
}
|
|
}
|
|
given .posted-at {
|
|
say 'Posted At: ', .Date.Str, ' ', .hh-mm-ss;
|
|
}
|
|
if .edited-at {
|
|
if $full {
|
|
say 'Edits: ';
|
|
for .edited-at.sort.reverse {
|
|
say ' * ', .Date.Str, ' ', .hh-mm-ss;
|
|
}
|
|
} else {
|
|
given .edited-at.sort[*-1] {
|
|
say 'Last Edited At: ', .Date.Str, ' ', .hh-mm-ss;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#| Add or remove a tag to a post
|
|
multi MAIN(
|
|
"post",
|
|
"tag",
|
|
#| The id of the post
|
|
Int $id,
|
|
#| The tag to add/remove
|
|
Str $tag,
|
|
#| remove the tag instead of adding it
|
|
Bool :$remove,
|
|
#| The path of the database directory
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
my $post = $db.posts{$id.Int};
|
|
if $remove {
|
|
die "Post did not have requested tag"
|
|
unless $tag ~~ any($post.tags);
|
|
my $index = $post.tags.first: $tag;
|
|
$post.tags.=grep(* ne $tag);
|
|
} else {
|
|
die "Post already had requested tag"
|
|
if $tag ~~ any($post.tags);
|
|
$post.tags.push: $tag;
|
|
}
|
|
$db.write: $db-dir;
|
|
}
|
|
|
|
#| Update the posted-at time on a post to the current time
|
|
multi MAIN(
|
|
"post",
|
|
"now",
|
|
#| The id of the post
|
|
Int $id,
|
|
#| The path of the database directory
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
){
|
|
my $db = read-db $db-dir;
|
|
my $post = $db.posts{$id.Int};
|
|
$post.posted-at = DateTime.now;
|
|
$db.write: $db-dir;
|
|
}
|
|
|
|
#| Set the source code link for a post
|
|
multi MAIN(
|
|
"post",
|
|
"source",
|
|
#| The id of the post
|
|
Int $id,
|
|
#| The path of the database directory
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
my $post = $db.posts{$id.Int};
|
|
|
|
say "Source Code Link:";
|
|
my $source-code = get;
|
|
|
|
$post.source-code = $source-code;
|
|
|
|
$db.write: $db-dir;
|
|
}
|
|
|
|
#| Create a new series
|
|
multi MAIN(
|
|
"series",
|
|
"new",
|
|
#| The path of the database file
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
say 'Series Title: ';
|
|
my $title = get;
|
|
say 'Series Description: ';
|
|
my $desc = get;
|
|
|
|
my $series = Series.new:
|
|
title => $title, desc => $desc;
|
|
my $id = $db.insert-series: $series;
|
|
say 'Series inserted with id ', $id;
|
|
|
|
$db.write: $db-dir;
|
|
}
|
|
|
|
|
|
#| Provide a table of series
|
|
multi MAIN(
|
|
"series",
|
|
"list",
|
|
#| The path of the database directory
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
#| The number of items to show on a single page
|
|
Int :$per-page = 10;
|
|
#| The page number to show
|
|
Int :$page = 1;
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
my @pages =
|
|
$db.series.sort(*.key).rotor($per-page, :partial);
|
|
my @page = @pages[$page - 1].flat;
|
|
my $table = Pretty::Table.new:
|
|
title => "Series (page $page/{@pages.elems})",
|
|
field-names => ["ID", "Title", "Desc"];
|
|
for @page -> $pair {
|
|
my $id = $pair.key;
|
|
my $series = $pair.value;
|
|
# TODO: Terminal aware truncation
|
|
my $title = do if $series.title.chars > 40 {
|
|
"{$series.title.substr(0, 50)}..."
|
|
} else {
|
|
$series.title
|
|
};
|
|
my $desc = do if $series.desc.chars > 40 {
|
|
"{$series.desc.substr(0, 50)}..."
|
|
} else {
|
|
$series.desc
|
|
};
|
|
$table.add-row: [$id, $title, $desc];
|
|
}
|
|
say $table;
|
|
}
|
|
|
|
#| Display the contents of a series
|
|
multi MAIN(
|
|
"series",
|
|
"info",
|
|
#| The id of the series to display
|
|
Int $id,
|
|
#| The path of the database directory
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
my $series = $db.series{$id.Int};
|
|
say 'Title: ', $series.title;
|
|
say 'Description:';
|
|
for $series.desc.lines -> $line {
|
|
say ' ', $line;
|
|
}
|
|
say 'Posts:';
|
|
for $series.post-ids -> $post-id {
|
|
my $post = $db.posts{$post-id};
|
|
say ' * ', $post-id, ': ', $post.title;
|
|
}
|
|
}
|
|
|
|
#| Add a post to a series
|
|
multi MAIN(
|
|
"series",
|
|
"add",
|
|
#| The id of the series to add to
|
|
Int $series-id,
|
|
#| The id of the post
|
|
Int $post-id,
|
|
#| The path of the database directory
|
|
IO::Path(Str) :$db-dir = $default-db-dir,
|
|
) {
|
|
my $db = read-db $db-dir;
|
|
my $series = $db.series{$series-id.Int};
|
|
$series.post-ids.push: $post-id.Int;
|
|
$db.write: $db-dir;
|
|
}
|