all repos — archive/homestead @ e58d6a1718b3358f91fbda10e2291ea4ec01d1e7

My future indieweb platform

feat: make taxonomies configurable

Add "tag" and "category" as defaults
Alan Pearce alan@alanpearce.eu
Sun, 18 Jun 2017 21:01:14 +0200
commit

e58d6a1718b3358f91fbda10e2291ea4ec01d1e7

parent

2fef943bf1c52e8d2be64521202936ca0f7358e3

4 files changed, 39 insertions(+), 15 deletions(-)

jump to
M config/default.tomlconfig/default.toml
@@ -2,4 +2,9 @@ [server] port = 3000
 
 [posts]
-folder = "../posts"+folder = "../posts"
+
+[taxonomies]
+tag = "tags"
+category = "categories"
+
M src/index.jssrc/index.js
@@ -37,14 +37,21 @@ post: post   })
 })
 
-const tags = Posts.toTags(posts)
-router.get('/tags/:tag', async function (ctx) {
-  ctx.assert(tags.has(ctx.params.tag), 404, 'Tag not found')
-  await ctx.render('tag', {
-    tag: ctx.params.tag,
-    posts: tags.get(ctx.params.tag)
+const taxonomies = Posts.taxonomise(config.taxonomies, posts)
+for (let [term, items] of taxonomies) {
+  router.get(`/${term}/:value`, async function (ctx) {
+    const value = ctx.params.value
+    ctx.assert(
+      items.has(ctx.params.value),
+      404,
+      `Could not find ${term} ${value}`
+    )
+    await ctx.render('tag', {
+      [term]: value,
+      posts: items.get(ctx.params.value)
+    })
   })
-})
+}
 
 app.use(router.routes()).use(router.allowedMethods())
 
M src/modules/posts.jssrc/modules/posts.js
@@ -56,21 +56,33 @@ .map(f => [getTitle(f), f])   )
 }
 
-function toTags (posts) {
-  const tags = new Map()
+function taxonomise (taxonomies, posts) {
+  const taxons = new Map(Object.keys(taxonomies).map(t => [t, new Map()]))
+
   for (let [, post] of posts) {
-    if (post.data.has('tags')) {
-      for (let tag of post.data.get('tags')) {
-        tags.set(tag, (tags.get(tag) || []).concat([post]))
+    for (let [singularName, pluralName] of Object.entries(taxonomies)) {
+      if (post.data.has(pluralName)) {
+        for (let term of post.data.get(pluralName)) {
+          const current = taxons.get(singularName).get(term)
+          taxons
+            .get(singularName)
+            .set(term, current ? current.concat(post) : [post])
+        }
       }
     }
   }
-  return tags
+
+  return taxons
+}
+
+function toTags (posts) {
+  return taxonomise({ tag: 'tags' }, posts).get('tag')
 }
 
 module.exports = {
   get,
   getFolder,
   toTags,
+  taxonomise,
   render
 }
M test/index.test.jstest/index.test.js
@@ -26,7 +26,7 @@ }) 
 test('tags', t => {
   return request(app.listen())
-    .get('/tags/a')
+    .get('/tag/a')
     .expect(200)
     .expect(/This is a test/)
     .then(() => t.pass())