aboutsummaryrefslogblamecommitdiffstats
path: root/server.js
blob: eea4695dcc9407e28091ca83b778c06e1980892c (plain) (tree)
1
2
3
4
5
6
7
8
9
10



                                   
                                               

                             


                                                  
                                        



                                                        





                                            
                             

                                 


                                 
 
                      
                            
                                                     
                                                        
                                                
                              
                              

















                                               
                                                                                                                                                                     














                                                     
                            
 



                                                           
                                           

                
                     










                                      
                    
               


                                                


          

 





                                             






                                                          










                                                                            

   








                                                                
                                    
                                  





                                              
                                                                          
       
   
 






                                                  
                             














                                                              

   
                                                                   








                                                                            




















                                                                
#!/usr/bin/env node
"use strict";

const express = require("express");
const sitemap = require("express-sitemap-xml");
const path = require("path");
const fs = require("fs");
const mit = require("markdown-it")({ html: true })
  .enable(["table"])
  .disable(["strikethrough"])
  .use(require("markdown-it-texmath"), {
    engine: require("katex"),
    delimiters: "gitlab",
    katexOptions: { macros: { "\\RR": "\\mathbb{R}" } },
  })
  .use(require("markdown-it-multimd-table"))
  .use(require("markdown-it-highlightjs"), {
    inline: true,
    auto: true,
    code: true,
  });
const spdy = require("spdy");
const helmet = require("helmet");
const morgan = require("morgan");
const model = require("./model");

model.dbInit();

const app = express();
app.disable("x-powered-by");
app.use(express.static(path.join(__dirname, "css")));
app.use(express.static(path.join(__dirname, "static")));
app.set("views", path.join(__dirname, "views"));
app.set("view engine", "ejs");
app.set("view engine", "pug");

app.use(helmet.crossOriginEmbedderPolicy());
app.use(helmet.crossOriginOpenerPolicy());
app.use(helmet.crossOriginResourcePolicy());
app.use(helmet.dnsPrefetchControl());
app.use(helmet.expectCt());
app.use(helmet.frameguard());
app.use(helmet.hidePoweredBy());
app.use(helmet.hsts());
app.use(helmet.ieNoOpen());
app.use(helmet.noSniff());
app.use(helmet.originAgentCluster());
app.use(helmet.permittedCrossDomainPolicies());
app.use(helmet.referrerPolicy());
app.use(helmet.xssFilter());
app.use((req, res, next) => {
  res.setHeader(
    "Permissions-Policy",
    "geolocation=(self),midi=(self),sync-xhr=(self),microphone=(self),camera=(self),magnetometer=(self),gyroscope=(self),fullscreen=(self),payment=(self),usb=(self)"
  );
  next();
});
app.use(
  helmet.contentSecurityPolicy({
    useDefaults: false,
    directives: {
      baseUri: ["self"],
      defaultSrc: ["self"],
      scriptSrc: ["none"],
      styleSrc: ["self", "https:", "unsafef-inline"],
    },
  })
);

app.use(morgan("combined"));

async function enumerateDir() {
  return await fs.readdirSync(path.join(__dirname, "mds"));
}

function renderAndSend_v2(req, res, slug) {
  model.blogPost
    .findOne(
      { slug: slug },
      {
        projection: {
          _id: 0,
          title: 0,
          teaser: 0,
        },
      }
    )
    .exec(function (err, blogPost) {
      if (err) return err;
      return res.render("index.ejs", {
        cache: true,
        data: {
          blogHttp: mit.render(blogPost.body),
          lastUpdatedAt: blogPost.lastUpdatedAt,
          keywords: blogPost.keywords,
        },
      });
    });
}

app.get("/health", (req, res) => {
  res.type("application/json");
  let response = { isOK: "True", error: "" };
  res.send(response);
});

app.get("/about", (req, res) => {
  res.type("text/html");
  res.sendFile(path.join(__dirname, "static/about.html"));
});

app.get("/archive", (req, res) => {
  res.type("text/html");
  model.blogPost
    .find({}, { _id: 0, body: 0, teaser: 0, keywords: 0, lastUpdatedAt: 0 })
    .exec(function (err, blogPosts) {
      if (err) return err;
      res.render("archive.ejs", {
        cache: true,
        data: {
          blogPosts: blogPosts,
        },
      });
    });
});

app.get("/robots.txt", (req, res) => {
  res.type("text/plain");
  let robots_txt = "Sitemap: http://blog.terminaldweller.com\n";
  robots_txt += "User-agent: *\n";
  robots_txt += "Disallow: \n";
  robots_txt += "Crawl-Delay: 20";
  res.send(robots_txt);
});

app.get("/rss/feed", (req, res) => {
  res.type("application/rss+xml");
  model.blogPost
    .find({})
    .sort("-lastUpdatedAt")
    .select("title slug lastUpdatedAt teaser")
    .exec(function (err, posts) {
      if (err) return err;
      return res.render("rss_feed_v2.pug", { cache: true, posts: posts });
    });
});

app.get("/posts/:postName", (req, res) => {
  if (req.params["postName"] == "") {
    res.write("nothing requested!");
  }
  renderAndSend_v2(req, res, req.params.postName);
});

app.get("/$", (req, res) => {
  model.blogPost
    .find({}, { projection: { _id: 0, title: 0, teaser: 0 } })
    .limit(1)
    .sort({ $natural: -1 })
    .exec(function (err, blogPost) {
      if (err) return err;
      return res.render("index.ejs", {
        cache: true,
        data: {
          blogHttp: mit.render(blogPost[0].body),
          lastUpdatedAt: blogPost[0].lastUpdatedAt,
          keywords: blogPost[0].keywords,
        },
      });
    });
});

app.use(sitemap(enumerateDir, "https://blog.terminaldweller.com"));

app.use((req, res) => {
  return res.status(404).send({ message: "Path" + req.url + "not found!" });
});

app.use((err, req, res) => {
  return res.status(500).send({ error: err });
});

if (process.env.SERVER_DEPLOYMENT_TYPE == "deployment") {
  spdy
    .createServer(
      {
        key: fs.readFileSync("/certs/privkey1.pem", "utf-8"),
        cert: fs.readFileSync("/certs/fullchain1.pem", "utf-8"),
      },
      app
    )
    .listen(process.env.SERVER_LISTEN_PORT || 9000);
} else if (process.env.SERVER_DEPLOYMENT_TYPE == "test") {
  spdy
    .createServer(
      {
        key: fs.readFileSync("/certs/server.key", "utf-8"),
        cert: fs.readFileSync("/certs/server.cert", "utf-8"),
      },
      app
    )
    .listen(process.env.SERVER_LISTEN_PORT || 9000);
}