app default endpoints
This commit is contained in:
Generated
+5
@@ -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",
|
||||||
|
|||||||
@@ -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
@@ -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
@@ -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
@@ -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
@@ -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}`;
|
||||||
|
|||||||
@@ -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, '')),
|
||||||
|
};
|
||||||
Reference in New Issue
Block a user