Compare commits
25 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 35fcdc7320 | |||
| dfb93b6764 | |||
| 6af4012522 | |||
| 1b91002c03 | |||
| bedd6a2953 | |||
| 52d37d56cd | |||
| fc7bc63c46 | |||
| 4397a76d9b | |||
| ddf964eb27 | |||
| 4b47276484 | |||
| a7fedb149f | |||
| ea95a285c9 | |||
| 0fde428806 | |||
| 8fc7ff1ca7 | |||
| ae4e2eb8d5 | |||
| 528e4be1fe | |||
| bd42883330 | |||
| b6ac0a73b4 | |||
| aebc3da5bc | |||
| 7a4a4f9006 | |||
| 1341aa5a56 | |||
| 5e05f250f4 | |||
| 6cf7be3afb | |||
| 6aceacad18 | |||
| a3a23be1c2 |
@@ -3,6 +3,7 @@
|
|||||||
/config.json
|
/config.json
|
||||||
/config.example.json
|
/config.example.json
|
||||||
/data
|
/data
|
||||||
|
/data/*
|
||||||
/test_data
|
/test_data
|
||||||
/access.log
|
/access.log
|
||||||
/error.log
|
/error.log
|
||||||
|
|||||||
@@ -125,6 +125,28 @@ Resources are located on the `data` folder and can be referenced as the root of
|
|||||||
/styles/main.css => data/styles/main.css
|
/styles/main.css => data/styles/main.css
|
||||||
```
|
```
|
||||||
|
|
||||||
|
In your template, the following data is sent :
|
||||||
|
|
||||||
|
* `info` (every pages)
|
||||||
|
* `title` : the blog's title as in the config
|
||||||
|
* `description` the blog's description as in the config
|
||||||
|
* `host` : the specified or guessed host with the protocol
|
||||||
|
* `version` : the GitBlog.md current running version
|
||||||
|
* `request` : the Express request object
|
||||||
|
* `config` : the content of the config
|
||||||
|
* `article` (article pages only)
|
||||||
|
* `title` : the full title
|
||||||
|
* `thumbnail` the URL path of the thumbnail
|
||||||
|
* `url` : the URL path for this article (with the title)
|
||||||
|
* `date` : a JS date
|
||||||
|
* `year`
|
||||||
|
* `month`
|
||||||
|
* `day`
|
||||||
|
* `path` : the URL path for the folder of the article (without the title)
|
||||||
|
* `realPath` : the system's path for the folder
|
||||||
|
* `escapedTitle` : the code with alphanumeric and underscore characters only
|
||||||
|
* `error` (error pages only) : the error code
|
||||||
|
|
||||||
#### 5. Create and init your git source
|
#### 5. Create and init your git source
|
||||||
|
|
||||||
You need to [create a new repository](https://github.com/new) on your favorite Git service.
|
You need to [create a new repository](https://github.com/new) on your favorite Git service.
|
||||||
@@ -132,7 +154,10 @@ You need to [create a new repository](https://github.com/new) on your favorite G
|
|||||||
```bash
|
```bash
|
||||||
#gitblog.md/
|
#gitblog.md/
|
||||||
cd data
|
cd data
|
||||||
|
git init
|
||||||
git remote add origin <url_of_your_repo.git>
|
git remote add origin <url_of_your_repo.git>
|
||||||
|
git add .
|
||||||
|
git commit -m "initial commit"
|
||||||
git push -u origin master
|
git push -u origin master
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -157,7 +182,7 @@ Here are the steps for Github, if you use another platform adapt it your way (he
|
|||||||
```json
|
```json
|
||||||
"webhook": {
|
"webhook": {
|
||||||
"endpoint": "/webhook",
|
"endpoint": "/webhook",
|
||||||
"secret": "sha1=<value>",
|
"secret": "<value>",
|
||||||
"signature_header": "X-Hub-Signature"
|
"signature_header": "X-Hub-Signature"
|
||||||
},
|
},
|
||||||
```
|
```
|
||||||
@@ -165,6 +190,14 @@ Here are the steps for Github, if you use another platform adapt it your way (he
|
|||||||
* Update your webhook on github to include the secret
|
* Update your webhook on github to include the secret
|
||||||
* Check if Github successfully reached the endpoint
|
* Check if Github successfully reached the endpoint
|
||||||
|
|
||||||
|
#### 8. Keep your server always up and running (optionnal)
|
||||||
|
|
||||||
|
This project `package.json` comes with a [nodemon](https://github.com/remy/nodemon) config.
|
||||||
|
|
||||||
|
After installing (`npm i -g nodemon`) you can then run the app with juste the `nodemon` command in the working directory.
|
||||||
|
|
||||||
|
With this method, you can do a simple `git pull` to update your server.
|
||||||
|
|
||||||
## Writing an article
|
## Writing an article
|
||||||
[back to top](#gitblog-md)
|
[back to top](#gitblog-md)
|
||||||
|
|
||||||
@@ -212,6 +245,9 @@ Any URL like `/year/month/day/anything/` will redirect to this article (and link
|
|||||||
|
|
||||||
* `node_port` (default: 3000)
|
* `node_port` (default: 3000)
|
||||||
the port the server is listening to
|
the port the server is listening to
|
||||||
|
* `host` (default: none)
|
||||||
|
if set (like `https://mywebsite.com`, it will be used as reference for creating links
|
||||||
|
by default, host is guessed based on first request
|
||||||
* `data_dir` (default: data)
|
* `data_dir` (default: data)
|
||||||
the directory where will be located the git repo with templates and articles
|
the directory where will be located the git repo with templates and articles
|
||||||
* `view_engine` (default: ejs)
|
* `view_engine` (default: ejs)
|
||||||
|
|||||||
+12
-1
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "gitblog.md",
|
"name": "gitblog.md",
|
||||||
"version": "1.1.5",
|
"version": "1.2.3",
|
||||||
"description": "A static blog using Markdown pulled from your git repository.",
|
"description": "A static blog using Markdown pulled from your git repository.",
|
||||||
"main": "src/server.js",
|
"main": "src/server.js",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
@@ -46,5 +46,16 @@
|
|||||||
"!src/postinstall.js",
|
"!src/postinstall.js",
|
||||||
"!src/lib/*.js"
|
"!src/lib/*.js"
|
||||||
]
|
]
|
||||||
|
},
|
||||||
|
"nodemonConfig": {
|
||||||
|
"verbose": true,
|
||||||
|
"ignore": [
|
||||||
|
"test/*",
|
||||||
|
"sample_data/*",
|
||||||
|
"data/*",
|
||||||
|
"uml/*",
|
||||||
|
"*.log",
|
||||||
|
"README.md"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,19 +1,8 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8">
|
<%- include('head'); %>
|
||||||
<title><%= info.title %> - Error <%= error %></title>
|
<title><%= info.title %> - Error <%= error %></title>
|
||||||
|
|
||||||
<meta name="twitter:card" content="summary_large_image">
|
|
||||||
<%- `<meta property="og:title" content="${info.title} - Home">` %>
|
|
||||||
<%- `<meta property="twitter:title" content="${info.title} - Home">` %>
|
|
||||||
<%- `<meta property="og:description" content="${info.description}">` %>
|
|
||||||
<%- `<meta property="twitter:description" content="${info.description}">` %>
|
|
||||||
<%- `<meta property="org:url" content="${info.host}/">` %>
|
|
||||||
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0, shrink-to-fit=no">
|
|
||||||
|
|
||||||
<link rel="stylesheet" type="text/css" href="/style.css">
|
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<main>
|
<main>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<hr>
|
<hr>
|
||||||
<footer>
|
<footer>
|
||||||
<small><a href="/rss">RSS feed</a> - @<%= new Date().getFullYear() %> - Made with <a
|
<small><a href="/rss">RSS feed</a> - <%= new Date().getFullYear() %> - Made with <a
|
||||||
href="https://github.com/klemek/gitblog.md">GitBlog.md</a> (v<%= info.version %>)
|
href="https://github.com/klemek/gitblog.md">GitBlog.md</a> (v<%= info.version %>)
|
||||||
</small>
|
</small>
|
||||||
</footer>
|
</footer>
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta http-equiv="content-type" content="text/html;charset=UTF-8">
|
||||||
|
<META NAME="ROBOTS" CONTENT="INDEX, FOLLOW">
|
||||||
|
|
||||||
|
<meta name="twitter:card" content="summary_large_image">
|
||||||
|
<%- `<meta property="og:description" content="${info.description}">` %>
|
||||||
|
<%- `<meta property="twitter:description" content="${info.description}">` %>
|
||||||
|
<% if(locals.article){ %>
|
||||||
|
<%- `<meta property="org:url" content="${info.host + article.url}">` %>
|
||||||
|
<%- `<meta property="og:title" content="${info.title} - ${article.title}">` %>
|
||||||
|
<%- `<meta property="twitter:title" content="${info.title} - ${article.title}">` %>
|
||||||
|
<% if (article.thumbnail) { %>
|
||||||
|
<%- `<meta property="og:image" content="${info.host}/${article.thumbnail}">` %>
|
||||||
|
<%- `<meta property="twitter:image" content="${info.host}/${article.thumbnail}">` %>
|
||||||
|
<% } %>
|
||||||
|
|
||||||
|
<link rel="stylesheet" type="text/css" href="/css/prism.css">
|
||||||
|
<% } else { %>
|
||||||
|
<%- `<meta property="org:url" content="${info.host}/">` %>
|
||||||
|
<%- `<meta property="og:title" content="${info.title} - Home">` %>
|
||||||
|
<%- `<meta property="twitter:title" content="${info.title} - Home">` %>
|
||||||
|
|
||||||
|
<%- `<meta property="description" content="${info.description}">` %>
|
||||||
|
<% } %>
|
||||||
|
|
||||||
|
<link rel="alternate" type="application/rss+xml" title="RSS feed" href="/rss"/>
|
||||||
|
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0, shrink-to-fit=no">
|
||||||
|
|
||||||
|
<link rel="stylesheet" type="text/css" href="/style.css">
|
||||||
@@ -1,19 +1,8 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8">
|
<%- include('head'); %>
|
||||||
<title><%= info.title %> - Home</title>
|
<title><%= info.title %> - Home</title>
|
||||||
|
|
||||||
<meta name="twitter:card" content="summary_large_image">
|
|
||||||
<%- `<meta property="og:title" content="${info.title} - Home">` %>
|
|
||||||
<%- `<meta property="twitter:title" content="${info.title} - Home">` %>
|
|
||||||
<%- `<meta property="og:description" content="${info.description}">` %>
|
|
||||||
<%- `<meta property="twitter:description" content="${info.description}">` %>
|
|
||||||
<%- `<meta property="org:url" content="${info.host}/">` %>
|
|
||||||
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0, shrink-to-fit=no">
|
|
||||||
|
|
||||||
<link rel="stylesheet" type="text/css" href="style.css">
|
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<main>
|
<main>
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ body, html {
|
|||||||
}
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
font: 14px/1.45 -apple-system, BlinkMacSystemFont, Segoe UI, sans-serif;
|
font: 15px sans-serif;
|
||||||
color: #111;
|
color: #111;
|
||||||
-webkit-text-size-adjust: none;
|
-webkit-text-size-adjust: none;
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
@@ -16,8 +16,8 @@ body {
|
|||||||
}
|
}
|
||||||
|
|
||||||
main {
|
main {
|
||||||
max-width: 75ch;
|
max-width: 42rem;
|
||||||
padding: 2ch;
|
padding: 2rem;
|
||||||
margin: auto;
|
margin: auto;
|
||||||
background-color: #F0F0F0;
|
background-color: #F0F0F0;
|
||||||
min-height: 100vh;
|
min-height: 100vh;
|
||||||
@@ -58,7 +58,7 @@ blockquote {
|
|||||||
border-left: 0.5em solid #ccc;
|
border-left: 0.5em solid #ccc;
|
||||||
padding-left: 1em;
|
padding-left: 1em;
|
||||||
margin: 0.25em 0;
|
margin: 0.25em 0;
|
||||||
color: #333;
|
color: #555;
|
||||||
}
|
}
|
||||||
|
|
||||||
blockquote > p {
|
blockquote > p {
|
||||||
@@ -108,7 +108,7 @@ main.article div.header a.link-home {
|
|||||||
line-height: 2.4;
|
line-height: 2.4;
|
||||||
}
|
}
|
||||||
|
|
||||||
main.article div.header h1, main.article div.header h2, div.article h3 {
|
main.article div.header h1, main.article div.header h2 {
|
||||||
margin-top: 0.85em;
|
margin-top: 0.85em;
|
||||||
margin-bottom: 0.25em;
|
margin-bottom: 0.25em;
|
||||||
font-size: 2em;
|
font-size: 2em;
|
||||||
@@ -123,11 +123,19 @@ main.article div.header span.time, div.article span.time {
|
|||||||
}
|
}
|
||||||
|
|
||||||
div.article {
|
div.article {
|
||||||
margin-left: 1em;
|
margin: 0 1em 1em 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
div.article h3 {
|
div.article h3 {
|
||||||
font-size: 1.3em;
|
font-size: 1.3em;
|
||||||
|
margin:0;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.article img{
|
||||||
|
max-width: 100%;
|
||||||
|
height: auto;
|
||||||
|
margin-right:1em;
|
||||||
|
margin-top:0.25em;
|
||||||
}
|
}
|
||||||
|
|
||||||
#text {
|
#text {
|
||||||
|
|||||||
@@ -1,24 +1,8 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8">
|
<%- include('head'); %>
|
||||||
<title><%= info.title %> - <%= article.title %></title>
|
<title><%= info.title %> - <%= article.title %></title>
|
||||||
|
|
||||||
<meta name="twitter:card" content="summary_large_image">
|
|
||||||
<%- `<meta property="og:title" content="${info.title} - ${article.title}">` %>
|
|
||||||
<%- `<meta property="twitter:title" content="${info.title} - ${article.title}">` %>
|
|
||||||
<%- `<meta property="og:description" content="${info.description}">` %>
|
|
||||||
<%- `<meta property="twitter:description" content="${info.description}">` %>
|
|
||||||
<%- `<meta property="org:url" content="${info.host + article.url}">` %>
|
|
||||||
<% if (article.thumbnail) { %>
|
|
||||||
<%- `<meta property="org:image" content="${info.host + article.thumbnail}">` %>
|
|
||||||
<%- `<meta property="twitter:image" content="${info.host + article.thumbnail}">` %>
|
|
||||||
<% } %>
|
|
||||||
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0, shrink-to-fit=no">
|
|
||||||
|
|
||||||
<link rel="stylesheet" type="text/css" href="/prism.css">
|
|
||||||
<link rel="stylesheet" type="text/css" href="/style.css">
|
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<main class="article">
|
<main class="article">
|
||||||
|
|||||||
+51
-43
@@ -26,6 +26,28 @@ const cons = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
module.exports = (config) => {
|
module.exports = (config) => {
|
||||||
|
/**
|
||||||
|
* Fetch articles from the data folder and send success as a response
|
||||||
|
* @param success
|
||||||
|
* @param error
|
||||||
|
*/
|
||||||
|
let reload;
|
||||||
|
/**
|
||||||
|
* Render the page with the view engine and catch errors
|
||||||
|
* @param req
|
||||||
|
* @param res
|
||||||
|
* @param vPath - path of the view
|
||||||
|
* @param data - data to pass to the view
|
||||||
|
* @param code - code to send along the page
|
||||||
|
*/
|
||||||
|
let render;
|
||||||
|
/**
|
||||||
|
* Show an error with the correct page
|
||||||
|
* @param req
|
||||||
|
* @param res
|
||||||
|
* @param code - error code
|
||||||
|
*/
|
||||||
|
let showError;
|
||||||
const fw = require('./file_walker')(config);
|
const fw = require('./file_walker')(config);
|
||||||
const renderer = require('./renderer')(config);
|
const renderer = require('./renderer')(config);
|
||||||
|
|
||||||
@@ -36,14 +58,9 @@ module.exports = (config) => {
|
|||||||
|
|
||||||
const articles = {};
|
const articles = {};
|
||||||
let lastRSS = '';
|
let lastRSS = '';
|
||||||
let host;
|
let host = config['host'];
|
||||||
|
|
||||||
/**
|
reload = (success, error) => {
|
||||||
* Fetch articles from the data folder and send success as a response
|
|
||||||
* @param success
|
|
||||||
* @param error
|
|
||||||
*/
|
|
||||||
const reload = (success, error) => {
|
|
||||||
fw.fetchArticles((err, dict) => {
|
fw.fetchArticles((err, dict) => {
|
||||||
if (err) {
|
if (err) {
|
||||||
console.error(cons.error, 'error loading articles : ' + err);
|
console.error(cons.error, 'error loading articles : ' + err);
|
||||||
@@ -65,42 +82,34 @@ module.exports = (config) => {
|
|||||||
if (config['test'])
|
if (config['test'])
|
||||||
app.reload = reload;
|
app.reload = reload;
|
||||||
|
|
||||||
/**
|
render = (req, res, vPath, data, code = 200) => {
|
||||||
* Render the page with the view engine and catch errors
|
|
||||||
* @param res
|
|
||||||
* @param vPath - path of the view
|
|
||||||
* @param data - data to pass to the view
|
|
||||||
* @param code - code to send along the page
|
|
||||||
*/
|
|
||||||
const render = (res, vPath, data, code = 200) => {
|
|
||||||
data.info = {
|
data.info = {
|
||||||
title: config['home']['title'],
|
title: config['home']['title'],
|
||||||
description: config['home']['description'],
|
description: config['home']['description'],
|
||||||
host: host,
|
host: host,
|
||||||
version: pjson.version
|
version: pjson.version,
|
||||||
|
request: req,
|
||||||
|
config: config
|
||||||
};
|
};
|
||||||
res.render(vPath, data, (err, html) => {
|
res.render(vPath, data, (err, html) => {
|
||||||
if (err) {
|
if (err && vPath !== path.join(config['data_dir'], config['home']['error'])) {
|
||||||
|
console.log(cons.error, `failed to render page ${vPath} : ${err}`);
|
||||||
|
showError(req, res, 500);
|
||||||
|
} else if (err) {
|
||||||
res.sendStatus(500);
|
res.sendStatus(500);
|
||||||
console.log(cons.error, `failed to render ${vPath} : ${err}`);
|
console.log(cons.error, `failed to render error page : ${err}`);
|
||||||
} else
|
} else
|
||||||
res.status(code).send(html);
|
res.status(code).send(html);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
showError = (req, res, code) => {
|
||||||
* Show an error with the correct page
|
|
||||||
* @param resPath - the page of the original error
|
|
||||||
* @param code - error code
|
|
||||||
* @param res
|
|
||||||
*/
|
|
||||||
const showError = (resPath, code, res) => {
|
|
||||||
const errorPath = path.join(config['data_dir'], config['home']['error']);
|
const errorPath = path.join(config['data_dir'], config['home']['error']);
|
||||||
fs.access(errorPath, fs.constants.R_OK, (err) => {
|
fs.access(errorPath, fs.constants.R_OK, (err) => {
|
||||||
if (err)
|
if (err)
|
||||||
res.sendStatus(code);
|
res.sendStatus(code);
|
||||||
else
|
else
|
||||||
render(res, errorPath, {error: code, path: resPath}, code);
|
render(req, res, errorPath, {error: code}, code);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -133,9 +142,9 @@ module.exports = (config) => {
|
|||||||
const homePath = path.join(config['data_dir'], config['home']['index']);
|
const homePath = path.join(config['data_dir'], config['home']['index']);
|
||||||
fs.access(homePath, fs.constants.R_OK, (err) => {
|
fs.access(homePath, fs.constants.R_OK, (err) => {
|
||||||
if (err)
|
if (err)
|
||||||
showError(req.path, 404, res);
|
showError(req, res, 404);
|
||||||
else
|
else
|
||||||
render(res, homePath, {articles: Object.values(articles).sort((a, b) => ('' + b.path).localeCompare(a.path))});
|
render(req, res, homePath, {articles: Object.values(articles).sort((a, b) => ('' + b.path).localeCompare(a.path))});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -146,23 +155,23 @@ module.exports = (config) => {
|
|||||||
const feed = new Rss({
|
const feed = new Rss({
|
||||||
'title': config['rss']['title'],
|
'title': config['rss']['title'],
|
||||||
'description': config['rss']['description'],
|
'description': config['rss']['description'],
|
||||||
'feed_url': 'http://' + req.headers.host + req.url,
|
'feed_url': host + req.url,
|
||||||
'site_url': 'http://' + req.headers.host
|
'site_url': host
|
||||||
});
|
});
|
||||||
Object.values(articles)
|
Object.values(articles)
|
||||||
.slice(0, config['rss']['length'])
|
.slice(0, config['rss']['length'])
|
||||||
.forEach((article) => {
|
.forEach((article) => {
|
||||||
feed.item({
|
feed.item({
|
||||||
title: article.title,
|
title: article.title,
|
||||||
url: 'http://' + req.headers.host + article.url,
|
url: host + article.url,
|
||||||
date: article.date
|
date: article.date
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
lastRSS = feed.xml();
|
lastRSS = feed.xml();
|
||||||
}
|
}
|
||||||
res.type('rss').send(lastRSS);
|
res.type(req.headers['user-agent'].match(/Mozilla/) ? 'text/xml' : 'rss').send(lastRSS);
|
||||||
} else {
|
} else {
|
||||||
showError(req.path, 404, res);
|
showError(req, res, 404);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -205,21 +214,21 @@ module.exports = (config) => {
|
|||||||
const articlePath = req.path.substr(1, 10);
|
const articlePath = req.path.substr(1, 10);
|
||||||
const article = articles[articlePath];
|
const article = articles[articlePath];
|
||||||
if (!article)
|
if (!article)
|
||||||
showError(req.path, 404, res);
|
showError(req, res, 404);
|
||||||
else {
|
else {
|
||||||
renderer.render(path.join(article.realPath, config['article']['index']), (err, html) => {
|
renderer.render(path.join(article.realPath, config['article']['index']), (err, html) => {
|
||||||
if (err) {
|
if (err) {
|
||||||
console.log(cons.error, `failed to render article ${req.path} : ${err}`);
|
console.log(cons.error, `failed to render article ${req.path} : ${err}`);
|
||||||
return showError(req.path, 500, res);
|
return showError(req, res, 500);
|
||||||
}
|
}
|
||||||
article.content = html;
|
article.content = html;
|
||||||
const templatePath = path.join(config['data_dir'], config['article']['template']);
|
const templatePath = path.join(config['data_dir'], config['article']['template']);
|
||||||
fs.access(templatePath, fs.constants.R_OK, (err) => {
|
fs.access(templatePath, fs.constants.R_OK, (err) => {
|
||||||
if (err) {
|
if (err) {
|
||||||
console.log(cons.error, `no template found at ${templatePath}`);
|
console.log(cons.error, `no template found at ${templatePath}`);
|
||||||
showError(req.path, 500, res);
|
showError(req, res, 500);
|
||||||
} else
|
} else
|
||||||
render(res, templatePath, {article: article});
|
render(req, res, templatePath, {article: article});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -229,18 +238,17 @@ module.exports = (config) => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// catch all hidden file type and return 404
|
// catch all hidden file type and return 404
|
||||||
app.get('*', (req, res, next) => {
|
config['home']['hidden'].forEach(pathMatcher => {
|
||||||
if (config['home']['hidden'].includes(path.extname(req.path)))
|
app.get(pathMatcher, (req, res) => {
|
||||||
showError(req.path, 404, res);
|
showError(req, res, 404);
|
||||||
else
|
});
|
||||||
next();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// serve all static files via get
|
// serve all static files via get
|
||||||
app.get('*', express.static(path.join(__dirname, '..', config['data_dir'])));
|
app.get('*', express.static(path.join(__dirname, '..', config['data_dir'])));
|
||||||
// catch express.static errors (mostly not found) by displaying 404
|
// catch express.static errors (mostly not found) by displaying 404
|
||||||
app.get('*', (req, res) => {
|
app.get('*', (req, res) => {
|
||||||
showError(req.path, 404, res);
|
showError(req, res, 404);
|
||||||
});
|
});
|
||||||
|
|
||||||
// catch all other methods and return 400
|
// catch all other methods and return 400
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
{
|
{
|
||||||
"node_port": 3000,
|
"node_port": 3000,
|
||||||
|
"host": "",
|
||||||
"data_dir": "data",
|
"data_dir": "data",
|
||||||
"view_engine": "ejs",
|
"view_engine": "ejs",
|
||||||
"access_log": "access.log",
|
"access_log": "access.log",
|
||||||
@@ -17,7 +18,8 @@
|
|||||||
"index": "index.ejs",
|
"index": "index.ejs",
|
||||||
"error": "error.ejs",
|
"error": "error.ejs",
|
||||||
"hidden": [
|
"hidden": [
|
||||||
".ejs"
|
"*.ejs",
|
||||||
|
"/.git*"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"article": {
|
"article": {
|
||||||
@@ -37,7 +39,7 @@
|
|||||||
"endpoint": "/webhook",
|
"endpoint": "/webhook",
|
||||||
"secret": "",
|
"secret": "",
|
||||||
"signature_header": "",
|
"signature_header": "",
|
||||||
"pull_command": "git pull"
|
"pull_command": "git pull origin master"
|
||||||
},
|
},
|
||||||
"showdown": {
|
"showdown": {
|
||||||
"parseImgDimensions": true,
|
"parseImgDimensions": true,
|
||||||
|
|||||||
+42
-9
@@ -16,16 +16,15 @@ config['data_dir'] = dataDir;
|
|||||||
config['webhook']['endpoint'] = '/webhooktest';
|
config['webhook']['endpoint'] = '/webhooktest';
|
||||||
config['rss']['endpoint'] = '/rsstest';
|
config['rss']['endpoint'] = '/rsstest';
|
||||||
config['rss']['length'] = 2;
|
config['rss']['length'] = 2;
|
||||||
config['home']['index'] = testIndex;
|
|
||||||
config['home']['error'] = testError;
|
config['home']['error'] = testError;
|
||||||
config['article']['template'] = testTemplate;
|
config['article']['template'] = testTemplate;
|
||||||
|
|
||||||
const app = require('../src/app')(config);
|
const app = require('../src/app')(config);
|
||||||
|
|
||||||
beforeEach((done, fail) => {
|
beforeEach((done, fail) => {
|
||||||
|
config['home']['index'] = testIndex;
|
||||||
config['data_dir'] = dataDir;
|
config['data_dir'] = dataDir;
|
||||||
config['article']['index'] = 'index.md';
|
config['article']['index'] = 'index.md';
|
||||||
config['home']['hidden'] = ['.ejs', '.test'];
|
|
||||||
config['access_log'] = '';
|
config['access_log'] = '';
|
||||||
config['error_log'] = '';
|
config['error_log'] = '';
|
||||||
config['modules']['rss'] = true;
|
config['modules']['rss'] = true;
|
||||||
@@ -93,20 +92,20 @@ describe('Test request logging', () => {
|
|||||||
|
|
||||||
describe('Test error logging', () => {
|
describe('Test error logging', () => {
|
||||||
test('test no log', (done) => {
|
test('test no log', (done) => {
|
||||||
config['home']['hidden'] = null;
|
config['home']['index'] = null;
|
||||||
request(app).get('/somefile.txt').then(() => {
|
request(app).get('/').then(() => {
|
||||||
expect(fs.existsSync(path.join(dataDir, 'error.log'))).toBe(false);
|
expect(fs.existsSync(path.join(dataDir, 'error.log'))).toBe(false);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
test('test null error ', (done) => {
|
test('test null error ', (done) => {
|
||||||
config['home']['hidden'] = null;
|
config['home']['index'] = null;
|
||||||
config['error_log'] = path.join(dataDir, 'error.log');
|
config['error_log'] = path.join(dataDir, 'error.log');
|
||||||
request(app).get('/somefile.txt').then(() => {
|
request(app).get('/').then(() => {
|
||||||
fs.readFile(path.join(dataDir, 'error.log'), {encoding: 'UTF-8'}, (err, data) => {
|
fs.readFile(path.join(dataDir, 'error.log'), {encoding: 'UTF-8'}, (err, data) => {
|
||||||
expect(err).toBeNull();
|
expect(err).toBeNull();
|
||||||
const start = data.split('\n').slice(0, 2).join('\n');
|
const start = data.split('\n').slice(0, 2).join('\n');
|
||||||
const expected = '500 GET /somefile.txt ' + new Date().toUTCString() + ' ::ffff:127.0.0.1\nTypeError: Cannot read property \'includes\' of null';
|
const expected = '500 GET / ' + new Date().toUTCString() + ' ::ffff:127.0.0.1\nTypeError [ERR_INVALID_ARG_TYPE]: The "path" argument must be of type string. Received type object';
|
||||||
expect(start).toBe(expected);
|
expect(start).toBe(expected);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
@@ -136,6 +135,23 @@ describe('Test root path', () => {
|
|||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
test('500 render error with page', (done) => {
|
||||||
|
fs.writeFileSync(path.join(dataDir, testIndex), 'articles <%= null.length %>');
|
||||||
|
fs.writeFileSync(path.join(dataDir, testError), 'error <%= error %>');
|
||||||
|
request(app).get('/').then((response) => {
|
||||||
|
expect(response.statusCode).toBe(500);
|
||||||
|
expect(response.text).toBe('error 500');
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('500 render error with failing page', (done) => {
|
||||||
|
fs.writeFileSync(path.join(dataDir, testIndex), 'articles <%= null.length %>');
|
||||||
|
fs.writeFileSync(path.join(dataDir, testError), 'error <%= null.error %>');
|
||||||
|
request(app).get('/').then((response) => {
|
||||||
|
expect(response.statusCode).toBe(500);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
test('200 no articles', (done) => {
|
test('200 no articles', (done) => {
|
||||||
fs.writeFileSync(path.join(dataDir, testIndex), 'articles <%= articles.length %>');
|
fs.writeFileSync(path.join(dataDir, testIndex), 'articles <%= articles.length %>');
|
||||||
request(app).get('/').then((response) => {
|
request(app).get('/').then((response) => {
|
||||||
@@ -175,11 +191,19 @@ describe('Test RSS feed', () => {
|
|||||||
test('200 empty rss', (done) => {
|
test('200 empty rss', (done) => {
|
||||||
request(app).get('/rsstest').then((response) => {
|
request(app).get('/rsstest').then((response) => {
|
||||||
expect(response.statusCode).toBe(200);
|
expect(response.statusCode).toBe(200);
|
||||||
|
expect(response.type).toBe('application/rss+xml');
|
||||||
expect(response.text.length).toBeGreaterThan(0);
|
expect(response.text.length).toBeGreaterThan(0);
|
||||||
expect(response.text.split('<item>').length).toBe(1);
|
expect(response.text.split('<item>').length).toBe(1);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
test('200 Mozilla fix', (done) => {
|
||||||
|
request(app).get('/rsstest').set('user-agent', 'Mozilla Firefox 64.0').then((response) => {
|
||||||
|
expect(response.statusCode).toBe(200);
|
||||||
|
expect(response.type).toBe('text/xml');
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
test('200 rss cache', (done) => {
|
test('200 rss cache', (done) => {
|
||||||
request(app).get('/rsstest').then(() => {
|
request(app).get('/rsstest').then(() => {
|
||||||
request(app).get('/rsstest').then((response) => {
|
request(app).get('/rsstest').then((response) => {
|
||||||
@@ -378,8 +402,17 @@ describe('Test static files', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
test('404 hidden file', (done) => {
|
test('404 hidden file', (done) => {
|
||||||
fs.writeFileSync(path.join(dataDir, 'somefile.test'), '');
|
utils.createEmptyDirs([path.join(dataDir, 'tmp')]);
|
||||||
request(app).get('/somefile.test').then((response) => {
|
fs.writeFileSync(path.join(dataDir, 'tmp', 'somefile.ejs'), '');
|
||||||
|
request(app).get('/tmp/somefile.ejs').then((response) => {
|
||||||
|
expect(response.statusCode).toBe(404);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('404 hidden folder', (done) => {
|
||||||
|
utils.createEmptyDirs([path.join(dataDir, '.git')]);
|
||||||
|
fs.writeFileSync(path.join(dataDir, '.git', 'file.txt'), '');
|
||||||
|
request(app).get('/.git/file.txt').then((response) => {
|
||||||
expect(response.statusCode).toBe(404);
|
expect(response.statusCode).toBe(404);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
|
|||||||
+1
-1
@@ -77,5 +77,5 @@ test('array fix', () => {
|
|||||||
fs.writeFileSync(configFile, '{"home":{"hidden":{}}}');
|
fs.writeFileSync(configFile, '{"home":{"hidden":{}}}');
|
||||||
const config = require('../src/config')();
|
const config = require('../src/config')();
|
||||||
expect(config).toBeDefined();
|
expect(config).toBeDefined();
|
||||||
expect(config['home']['hidden']).toEqual(['.ejs']);
|
expect(config['home']['hidden']).toEqual(['*.ejs', '/.git*']);
|
||||||
});
|
});
|
||||||
Reference in New Issue
Block a user