app default endpoints

This commit is contained in:
Clément GOUIN
2019-06-20 10:07:22 +02:00
parent 7710eaad25
commit fe6cca6453
7 changed files with 193 additions and 48 deletions
+5
View File
@@ -2869,6 +2869,11 @@
"resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
"integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0="
}, },
"ejs": {
"version": "2.6.2",
"resolved": "https://registry.npmjs.org/ejs/-/ejs-2.6.2.tgz",
"integrity": "sha512-PcW2a0tyTuPHz3tWyYqtK6r1fZ3gp+3Sop8Ph+ZYN81Ob5rwmbHEzaqs10N3BEsaGTkh/ooniXK+WwszGlc2+Q=="
},
"electron-to-chromium": { "electron-to-chromium": {
"version": "1.3.165", "version": "1.3.165",
"resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.165.tgz", "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.165.tgz",
+1
View File
@@ -5,6 +5,7 @@
"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": {
"ejs": "^2.6.2",
"express": "^4.17.1", "express": "^4.17.1",
"ncp": "^2.0.0", "ncp": "^2.0.0",
"showdown": "^1.9.0", "showdown": "^1.9.0",
+48 -4
View File
@@ -1,5 +1,7 @@
const express = require('express'); const express = require('express');
const app = express(); const app = express();
const fs = require('fs');
const path = require('path');
const cons = { const cons = {
ok: '\x1b[32m✔\x1b[0m %s', ok: '\x1b[32m✔\x1b[0m %s',
@@ -10,8 +12,17 @@ const cons = {
module.exports = (config) => { module.exports = (config) => {
const fw = require('./file_walker')(config); const fw = require('./file_walker')(config);
app.set('view engine', config['view_engine']);
app.set('views', path.join(__dirname, '..'));
const articles = []; const articles = [];
const log = (status, msg) => {
if (config['test'])
return;
console.log(status, msg);
};
const reload = (callback) => { const reload = (callback) => {
fw.fetchArticles((err, list) => { fw.fetchArticles((err, list) => {
if (err) { if (err) {
@@ -20,22 +31,55 @@ module.exports = (config) => {
} }
articles.splice(0, articles.length, ...list); articles.splice(0, articles.length, ...list);
if (articles.length > 0) if (articles.length > 0)
console.log(cons.ok, `loaded ${articles.length} article${articles.length > 1 ? 's' : ''}`); log(cons.ok, `loaded ${articles.length} article${articles.length > 1 ? 's' : ''}`);
else else
console.log(cons.warn, `no articles loaded, check your configuration`); log(cons.warn, `no articles loaded, check your configuration`);
callback(true); callback(true);
}); });
}; };
const render = (res, path, data, code = 200) => {
res.render(path, data, (err, html) => {
if (err) {
res.sendStatus(500);
log(cons.error, `failed to render ${path} : ${err}`);
} else
res.status(code).send(html);
});
};
const showError = (path, code, res) => {
const errorPath = `${config['data_dir']}/${config['home']['error']}`;
if (fs.existsSync(errorPath))
render(res, errorPath, {error: code, path: path}, code);
else
res.sendStatus(code);
};
app.get('/', (req, res) => { app.get('/', (req, res) => {
res.status(200).send('Hello World!'); const homePath = `${config['data_dir']}/${config['home']['index']}`;
if (fs.existsSync(homePath))
render(res, homePath, {articles: articles});
else {
showError(req.path, 404, res);
}
});
app.get('*', express.static(config['data_dir']));
app.get('*', (req, res) => {
showError(req.path, 404, res);
});
app.all('*', (req, res) => {
res.status(400).send('bad request');
}); });
app.start = () => { app.start = () => {
reload((res) => { reload((res) => {
if (res) if (res)
app.listen(config['node_port'], () => { app.listen(config['node_port'], () => {
console.log(cons.ok, `gitblog.md server listening on port ${config['node_port']}`); log(cons.ok, `gitblog.md server listening on port ${config['node_port']}`);
}); });
}); });
}; };
+14 -12
View File
@@ -1,25 +1,27 @@
{ {
"node_port": 3000, "node_port": 3000,
"data_dir": "data", "data_dir": "data",
"modules" : { "view_engine": "ejs",
"plantuml" : false, "modules": {
"plantuml": false,
"rss": true, "rss": true,
"webhook": true "webhook": true
}, },
"home" : { "home": {
"index" : "index.ejs" "index": "index.ejs",
"error": "error.ejs"
}, },
"article" : { "article": {
"index" : "index.md", "index": "index.md",
"thumbnail_tag" : "thumbnail", "thumbnail_tag": "thumbnail",
"default_title": "Untitled", "default_title": "Untitled",
"default_thumbnail" : null "default_thumbnail": null
}, },
"rss" : { "rss": {
"endpoint" : "/rss", "endpoint": "/rss",
"length" : 10 "length": 10
}, },
"webhook" : { "webhook": {
"endpoint": "/webhook", "endpoint": "/webhook",
"secretFile": "git_secret" "secretFile": "git_secret"
} }
+90 -2
View File
@@ -1,15 +1,103 @@
/* jshint -W117 */ /* jshint -W117 */
const request = require('supertest'); const request = require('supertest');
const app = require('../src/app')({ const fs = require('fs');
const utils = require('./test_utils');
const dataDir = './test_data';
const testIndex = 'testindex.ejs';
const testError = 'testerror.ejs';
const config = {
'test': true,
'data_dir': dataDir,
'view_engine': 'ejs',
'home': {
'index': testIndex,
'error': testError
}
};
const app = require('../src/app')(config);
beforeEach(() => {
utils.deleteFolderSync(dataDir);
fs.mkdirSync(dataDir);
});
afterAll(() => {
if (fs.existsSync(dataDir)) {
utils.deleteFolderSync(dataDir);
}
}); });
describe('Test root path', () => { describe('Test root path', () => {
test('GET / 200', done => { test('404 no index no error', done => {
request(app).get('/').then(response => {
expect(response.statusCode).toBe(404);
done();
});
});
test('404 no index but error page', done => {
fs.writeFileSync(`${dataDir}/${testError}`, 'error <%= error %> at <%= path %>');
request(app).get('/').then(response => {
expect(response.statusCode).toBe(404);
expect(response.text).toBe('error 404 at /');
done();
});
});
test('200 index page', done => {
fs.writeFileSync(`${dataDir}/${testIndex}`, 'hello there');
request(app).get('/').then(response => { request(app).get('/').then(response => {
expect(response.statusCode).toBe(200); expect(response.statusCode).toBe(200);
expect(response.text).toBe('hello there');
done();
});
});
//TODO test articles list
});
describe('Test static files', () => {
test('404 invalid file no error page', done => {
request(app).get('/somefile.txt').then(response => {
expect(response.statusCode).toBe(404);
done();
});
});
test('404 invalid file but error page', done => {
fs.writeFileSync(`${dataDir}/${testError}`, 'error <%= error %> at <%= path %>');
request(app).get('/somefile.txt').then(response => {
expect(response.statusCode).toBe(404);
expect(response.text).toBe('error 404 at /somefile.txt');
done();
});
});
test('200 valid file', done => {
fs.writeFileSync(`${dataDir}/somefile.txt`, 'filecontent');
request(app).get('/somefile.txt').then(response => {
expect(response.statusCode).toBe(200);
expect(response.text).toBe('filecontent');
done(); done();
}); });
}); });
}); });
describe('Test other requests', () => {
test('400 POST', done => {
request(app).post('/').then(response => {
expect(response.statusCode).toBe(400);
done();
});
});
test('400 PUT', done => {
request(app).put('/').then(response => {
expect(response.statusCode).toBe(400);
done();
});
});
test('400 DELETE', done => {
request(app).delete('/').then(response => {
expect(response.statusCode).toBe(400);
done();
});
});
});
+12 -26
View File
@@ -1,5 +1,6 @@
/* jshint -W117 */ /* jshint -W117 */
const fs = require('fs'); const fs = require('fs');
const utils = require('./test_utils');
const dataDir = './test_data'; const dataDir = './test_data';
const testIndex = 'testindex.md'; const testIndex = 'testindex.md';
@@ -17,32 +18,17 @@ const config = {
const fw = require('../src/file_walker')(config); const fw = require('../src/file_walker')(config);
const deleteFolderSync = (path) => {
if (!fs.existsSync(path))
return;
fs.readdirSync(path, {withFileTypes: true}).forEach((item) => {
if (item.isDirectory())
deleteFolderSync(`${path}/${item.name}`);
else
fs.unlinkSync(`${path}/${item.name}`);
});
fs.rmdirSync(path);
};
beforeEach(() => { beforeEach(() => {
deleteFolderSync(dataDir); utils.deleteFolderSync(dataDir);
fs.mkdirSync(dataDir); fs.mkdirSync(dataDir);
}); });
afterAll(() => { afterAll(() => {
if (fs.existsSync(dataDir)) { if (fs.existsSync(dataDir)) {
deleteFolderSync(dataDir); utils.deleteFolderSync(dataDir);
} }
}); });
const createEmptyDirs = list => list.forEach(path => fs.mkdirSync(path, {recursive: true}));
const createEmptyFiles = list => list.forEach(file => fs.writeFileSync(file, ''));
describe('Test function fileTree', () => { describe('Test function fileTree', () => {
test('empty root', (done) => { test('empty root', (done) => {
fw.fileTree(dataDir, (err, list) => { fw.fileTree(dataDir, (err, list) => {
@@ -53,7 +39,7 @@ describe('Test function fileTree', () => {
}); });
}); });
test('empty folders', (done) => { test('empty folders', (done) => {
createEmptyDirs([ utils.createEmptyDirs([
`${dataDir}/test/test`, `${dataDir}/test/test`,
`${dataDir}/test/test2`, `${dataDir}/test/test2`,
`${dataDir}/test2` `${dataDir}/test2`
@@ -70,7 +56,7 @@ describe('Test function fileTree', () => {
`${dataDir}/f1.txt`, `${dataDir}/f1.txt`,
`${dataDir}/f2.txt` `${dataDir}/f2.txt`
]; ];
createEmptyFiles(fileList); utils.createEmptyFiles(fileList);
fw.fileTree(dataDir, (err, list) => { fw.fileTree(dataDir, (err, list) => {
expect(err).toBeNull(); expect(err).toBeNull();
expect(list).toBeDefined(); expect(list).toBeDefined();
@@ -80,7 +66,7 @@ describe('Test function fileTree', () => {
}); });
}); });
test('nested files', (done) => { test('nested files', (done) => {
createEmptyDirs([ utils.createEmptyDirs([
`${dataDir}/test/test`, `${dataDir}/test/test`,
`${dataDir}/test2` `${dataDir}/test2`
]); ]);
@@ -90,7 +76,7 @@ describe('Test function fileTree', () => {
`${dataDir}/test/test/f3.txt`, `${dataDir}/test/test/f3.txt`,
`${dataDir}/test2/f4.txt` `${dataDir}/test2/f4.txt`
]; ];
createEmptyFiles(fileList); utils.createEmptyFiles(fileList);
fw.fileTree(dataDir, (err, list) => { fw.fileTree(dataDir, (err, list) => {
expect(err).toBeNull(); expect(err).toBeNull();
expect(list).toBeDefined(); expect(list).toBeDefined();
@@ -212,11 +198,11 @@ describe('Test article fetching', () => {
}); });
}); });
test('misplaced index file', (done) => { test('misplaced index file', (done) => {
createEmptyDirs([ utils.createEmptyDirs([
`${dataDir}/test/test`, `${dataDir}/test/test`,
`${dataDir}/2019/05/05`, `${dataDir}/2019/05/05`,
]); ]);
createEmptyFiles([ utils.createEmptyFiles([
`${dataDir}/${testIndex}`, `${dataDir}/${testIndex}`,
`${dataDir}/test/test/${testIndex}`, `${dataDir}/test/test/${testIndex}`,
`${dataDir}/2019/05/${testIndex}`, `${dataDir}/2019/05/${testIndex}`,
@@ -229,11 +215,11 @@ describe('Test article fetching', () => {
}); });
}); });
test('empty index file', (done) => { test('empty index file', (done) => {
createEmptyDirs([ utils.createEmptyDirs([
`${dataDir}/2019/05/05`, `${dataDir}/2019/05/05`,
]); ]);
const file = `${dataDir}/2019/05/05/${testIndex}`; const file = `${dataDir}/2019/05/05/${testIndex}`;
createEmptyFiles([file]); utils.createEmptyFiles([file]);
fw.fetchArticles((err, list) => { fw.fetchArticles((err, list) => {
expect(err).toBeNull(); expect(err).toBeNull();
expect(list).toBeDefined(); expect(list).toBeDefined();
@@ -250,7 +236,7 @@ describe('Test article fetching', () => {
}); });
}); });
test('correct index file', (done) => { test('correct index file', (done) => {
createEmptyDirs([ utils.createEmptyDirs([
`${dataDir}/2019/05/05`, `${dataDir}/2019/05/05`,
]); ]);
const file = `${dataDir}/2019/05/05/${testIndex}`; const file = `${dataDir}/2019/05/05/${testIndex}`;
+19
View File
@@ -0,0 +1,19 @@
const fs = require('fs');
const deleteFolderSync = (path) => {
if (!fs.existsSync(path))
return;
fs.readdirSync(path, {withFileTypes: true}).forEach((item) => {
if (item.isDirectory())
deleteFolderSync(`${path}/${item.name}`);
else
fs.unlinkSync(`${path}/${item.name}`);
});
fs.rmdirSync(path);
};
module.exports = {
deleteFolderSync: deleteFolderSync,
createEmptyDirs: list => list.forEach(path => fs.mkdirSync(path, {recursive: true})),
createEmptyFiles: list => list.forEach(file => fs.writeFileSync(file, '')),
};