RSS feed in app
This commit is contained in:
Generated
+30
-8
@@ -8296,6 +8296,30 @@
|
|||||||
"glob": "^7.1.3"
|
"glob": "^7.1.3"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"rss": {
|
||||||
|
"version": "1.2.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/rss/-/rss-1.2.2.tgz",
|
||||||
|
"integrity": "sha1-UKFpiHYTgTOnT5oF0r3I240nqSE=",
|
||||||
|
"requires": {
|
||||||
|
"mime-types": "2.1.13",
|
||||||
|
"xml": "1.0.1"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"mime-db": {
|
||||||
|
"version": "1.25.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.25.0.tgz",
|
||||||
|
"integrity": "sha1-wY29fHOl2/b0SgJNwNFloeexw5I="
|
||||||
|
},
|
||||||
|
"mime-types": {
|
||||||
|
"version": "2.1.13",
|
||||||
|
"resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.13.tgz",
|
||||||
|
"integrity": "sha1-4HqqnGxrmnyjASxpADrSWjnpKog=",
|
||||||
|
"requires": {
|
||||||
|
"mime-db": "~1.25.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"rsvp": {
|
"rsvp": {
|
||||||
"version": "4.8.5",
|
"version": "4.8.5",
|
||||||
"resolved": "https://registry.npmjs.org/rsvp/-/rsvp-4.8.5.tgz",
|
"resolved": "https://registry.npmjs.org/rsvp/-/rsvp-4.8.5.tgz",
|
||||||
@@ -8670,7 +8694,8 @@
|
|||||||
"sax": {
|
"sax": {
|
||||||
"version": "1.2.4",
|
"version": "1.2.4",
|
||||||
"resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz",
|
"resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz",
|
||||||
"integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw=="
|
"integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==",
|
||||||
|
"dev": true
|
||||||
},
|
},
|
||||||
"semver": {
|
"semver": {
|
||||||
"version": "5.7.0",
|
"version": "5.7.0",
|
||||||
@@ -9669,13 +9694,10 @@
|
|||||||
"async-limiter": "~1.0.0"
|
"async-limiter": "~1.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"xml-js": {
|
"xml": {
|
||||||
"version": "1.6.11",
|
"version": "1.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/xml-js/-/xml-js-1.6.11.tgz",
|
"resolved": "https://registry.npmjs.org/xml/-/xml-1.0.1.tgz",
|
||||||
"integrity": "sha512-7rVi2KMfwfWFl+GpPg6m80IVMWXLRjO+PxTq7V2CDhoGak0wzYzFgUY2m4XJ47OGdXd8eLE8EmwfAmdjw7lC1g==",
|
"integrity": "sha1-eLpyAgApxbyHuKgaPPzXS0ovweU="
|
||||||
"requires": {
|
|
||||||
"sax": "^1.2.4"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"xml-name-validator": {
|
"xml-name-validator": {
|
||||||
"version": "3.0.0",
|
"version": "3.0.0",
|
||||||
|
|||||||
+2
-2
@@ -8,8 +8,8 @@
|
|||||||
"ejs": "^2.6.2",
|
"ejs": "^2.6.2",
|
||||||
"express": "^4.17.1",
|
"express": "^4.17.1",
|
||||||
"ncp": "^2.0.0",
|
"ncp": "^2.0.0",
|
||||||
"showdown": "^1.9.0",
|
"rss": "^1.2.2",
|
||||||
"xml-js": "^1.6.11"
|
"showdown": "^1.9.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"babel-cli": "^6.26.0",
|
"babel-cli": "^6.26.0",
|
||||||
|
|||||||
+33
-1
@@ -2,6 +2,7 @@ const express = require('express');
|
|||||||
const app = express();
|
const app = express();
|
||||||
const fs = require('fs');
|
const fs = require('fs');
|
||||||
const path = require('path');
|
const path = require('path');
|
||||||
|
const Rss = require('rss');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Terminal colors and symbols to display status messages
|
* Terminal colors and symbols to display status messages
|
||||||
@@ -23,6 +24,7 @@ module.exports = (config) => {
|
|||||||
app.set('views', path.join(__dirname, '..'));
|
app.set('views', path.join(__dirname, '..'));
|
||||||
|
|
||||||
const articles = {};
|
const articles = {};
|
||||||
|
let lastRSS = '';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fetch articles from the data folder and send success as a response
|
* Fetch articles from the data folder and send success as a response
|
||||||
@@ -41,6 +43,9 @@ module.exports = (config) => {
|
|||||||
console.log(cons.ok, `loaded ${nb} article${nb > 1 ? 's' : ''}`);
|
console.log(cons.ok, `loaded ${nb} article${nb > 1 ? 's' : ''}`);
|
||||||
else
|
else
|
||||||
console.log(cons.warn, `no articles loaded, check your configuration`);
|
console.log(cons.warn, `no articles loaded, check your configuration`);
|
||||||
|
|
||||||
|
lastRSS = '';
|
||||||
|
|
||||||
callback(true);
|
callback(true);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
@@ -87,10 +92,37 @@ module.exports = (config) => {
|
|||||||
if (err)
|
if (err)
|
||||||
showError(req.path, 404, res);
|
showError(req.path, 404, res);
|
||||||
else
|
else
|
||||||
render(res, homePath, {articles: Object.values(articles)});
|
render(res, homePath, {articles: Object.values(articles).sort((a, b) => ('' + b.path).localeCompare(a.path))});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
//RSS endpoint
|
||||||
|
app.get(config['rss']['endpoint'], (req, res) => {
|
||||||
|
if (config['modules']['rss']) {
|
||||||
|
if (!lastRSS) {
|
||||||
|
const feed = new Rss({
|
||||||
|
'title': config['rss']['title'],
|
||||||
|
'description': config['rss']['description'],
|
||||||
|
'feed_url': 'http://' + req.headers.host + req.url,
|
||||||
|
'site_url': 'http://' + req.headers.host
|
||||||
|
});
|
||||||
|
Object.values(articles)
|
||||||
|
.slice(0, config['rss']['length'])
|
||||||
|
.forEach((article) => {
|
||||||
|
feed.item({
|
||||||
|
title: article.title,
|
||||||
|
url: 'http://' + req.headers.host + article.url,
|
||||||
|
date: article.date
|
||||||
|
});
|
||||||
|
});
|
||||||
|
lastRSS = feed.xml();
|
||||||
|
}
|
||||||
|
res.type('rss').send(lastRSS);
|
||||||
|
} else {
|
||||||
|
showError(req.path, 404, res);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
// catch all article urls and render them
|
// catch all article urls and render them
|
||||||
app.get('*', (req, res, next) => {
|
app.get('*', (req, res, next) => {
|
||||||
if (/^\/\d{4}\/\d{2}\/\d{2}\/(\w*\/)?$/.test(req.path)) {
|
if (/^\/\d{4}\/\d{2}\/\d{2}\/(\w*\/)?$/.test(req.path)) {
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"node_port": 3000,
|
"node_port": 3000,
|
||||||
"data_dir": "data",
|
"data_dir": "data",
|
||||||
"view_engine": "ejs",
|
"view_engine": "ejs",
|
||||||
"loopback_url": "http://localhost:3000",
|
"language": "en-us",
|
||||||
"modules": {
|
"modules": {
|
||||||
"plantuml": false,
|
"plantuml": false,
|
||||||
"rss": true,
|
"rss": true,
|
||||||
|
|||||||
@@ -87,6 +87,7 @@ module.exports = (config) => {
|
|||||||
day: parseInt(matches[3])
|
day: parseInt(matches[3])
|
||||||
};
|
};
|
||||||
article.date = new Date(article.year, article.month, article.day);
|
article.date = new Date(article.year, article.month, article.day);
|
||||||
|
article.date.setUTCHours(0);
|
||||||
remaining++;
|
remaining++;
|
||||||
readIndexFile(path.join(article.realPath, config['article']['index']), config['article']['thumbnail_tag'], (err, info) => {
|
readIndexFile(path.join(article.realPath, config['article']['index']), config['article']['thumbnail_tag'], (err, info) => {
|
||||||
if (err)
|
if (err)
|
||||||
|
|||||||
-40
@@ -1,40 +0,0 @@
|
|||||||
const convert = require('xml-js');
|
|
||||||
|
|
||||||
const mapArticle = (url, article) => {
|
|
||||||
return {
|
|
||||||
'title': {'_text': article.title},
|
|
||||||
'link': {'_text': (url + article.url).replace(/([^:])\/\//g, '$1/')},
|
|
||||||
'pubDate': {'_text': article.date.toString()},
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = (config) => {
|
|
||||||
return {
|
|
||||||
get: (dict) => {
|
|
||||||
const items = Object.values(dict)
|
|
||||||
.sort((a, b) => ('' + b.path).localeCompare(a.path))
|
|
||||||
.slice(0, config['rss']['length']);
|
|
||||||
const data = {
|
|
||||||
'_declaration': {
|
|
||||||
'_attributes': {
|
|
||||||
'version': '1.0',
|
|
||||||
'encoding': 'UTF-8'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
'rss': {
|
|
||||||
'_attributes': {
|
|
||||||
'version': '2.0'
|
|
||||||
},
|
|
||||||
'title': {'_text': config['rss']['title']},
|
|
||||||
'description': {'_text': config['rss']['description']},
|
|
||||||
'link': {'_text': config['loopback_url']},
|
|
||||||
'lastBuildDate': {'_text': new Date().toString()},
|
|
||||||
'lastPubDate': {'_text': new Date().toString()},
|
|
||||||
'ttl': {'_text': '60'},
|
|
||||||
'item': items.map((a) => mapArticle(config['loopback_url'], a))
|
|
||||||
}
|
|
||||||
};
|
|
||||||
return convert.js2xml(data, {compact: true});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
};
|
|
||||||
+74
-19
@@ -9,30 +9,26 @@ const testIndex = 'testindex.ejs';
|
|||||||
const testError = 'testerror.ejs';
|
const testError = 'testerror.ejs';
|
||||||
const testTemplate = 'testtemplate.ejs';
|
const testTemplate = 'testtemplate.ejs';
|
||||||
|
|
||||||
const config = {
|
const config = require('../src/config')();
|
||||||
'test': true,
|
|
||||||
'data_dir': dataDir,
|
config['test'] = true;
|
||||||
'view_engine': 'ejs',
|
config['data_dir'] = dataDir;
|
||||||
'home': {
|
config['home']['index'] = testIndex;
|
||||||
'index': testIndex,
|
config['home']['error'] = testError;
|
||||||
'error': testError,
|
config['article']['template'] = testTemplate;
|
||||||
'hidden': ['.ejs', '.test']
|
config['home']['hidden'].push('.test');
|
||||||
},
|
config['rss']['endpoint'] = '/rsstest';
|
||||||
'article': {
|
config['rss']['length'] = 2;
|
||||||
'index': 'index.md',
|
|
||||||
'template': testTemplate,
|
|
||||||
'thumbnail_tag': 'thumbnail',
|
|
||||||
'default_title': 'Untitled',
|
|
||||||
'default_thumbnail': null
|
|
||||||
},
|
|
||||||
'showdown': {}
|
|
||||||
};
|
|
||||||
|
|
||||||
const app = require('../src/app')(config);
|
const app = require('../src/app')(config);
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach((done) => {
|
||||||
utils.deleteFolderSync(dataDir);
|
utils.deleteFolderSync(dataDir);
|
||||||
fs.mkdirSync(dataDir);
|
fs.mkdirSync(dataDir);
|
||||||
|
app.reload((res) => {
|
||||||
|
expect(res).toBe(true);
|
||||||
|
done();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
afterAll(() => {
|
afterAll(() => {
|
||||||
@@ -86,6 +82,65 @@ describe('Test root path', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('Test RSS feed', () => {
|
||||||
|
test('404 rss deactivated', (done) => {
|
||||||
|
config['modules']['rss'] = false;
|
||||||
|
request(app).get('/rsstest').then((response) => {
|
||||||
|
expect(response.statusCode).toBe(404);
|
||||||
|
config['modules']['rss'] = true;
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('200 empty rss', (done) => {
|
||||||
|
request(app).get('/rsstest').then((response) => {
|
||||||
|
expect(response.statusCode).toBe(200);
|
||||||
|
expect(response.text.length).toBeGreaterThan(0);
|
||||||
|
expect(response.text.split('<item>').length).toBe(1);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('200 2 rss items', (done) => {
|
||||||
|
utils.createEmptyDirs([
|
||||||
|
path.join(dataDir, '2019', '05', '05'),
|
||||||
|
path.join(dataDir, '2018', '05', '05')
|
||||||
|
]);
|
||||||
|
utils.createEmptyFiles([
|
||||||
|
path.join(dataDir, '2019', '05', '05', 'index.md'),
|
||||||
|
path.join(dataDir, '2018', '05', '05', 'index.md')
|
||||||
|
]);
|
||||||
|
app.reload((res) => {
|
||||||
|
expect(res).toBe(true);
|
||||||
|
request(app).get('/rsstest').then((response) => {
|
||||||
|
expect(response.statusCode).toBe(200);
|
||||||
|
expect(response.text.length).toBeGreaterThan(0);
|
||||||
|
expect(response.text.split('<item>').length).toBe(3);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('200 max rss items', (done) => {
|
||||||
|
utils.createEmptyDirs([
|
||||||
|
path.join(dataDir, '2019', '05', '05'),
|
||||||
|
path.join(dataDir, '2018', '05', '05'),
|
||||||
|
path.join(dataDir, '2017', '05', '05')
|
||||||
|
]);
|
||||||
|
utils.createEmptyFiles([
|
||||||
|
path.join(dataDir, '2019', '05', '05', 'index.md'),
|
||||||
|
path.join(dataDir, '2018', '05', '05', 'index.md'),
|
||||||
|
path.join(dataDir, '2017', '05', '05', 'index.md')
|
||||||
|
]);
|
||||||
|
app.reload((res) => {
|
||||||
|
expect(res).toBe(true);
|
||||||
|
request(app).get('/rsstest').then((response) => {
|
||||||
|
expect(response.statusCode).toBe(200);
|
||||||
|
expect(response.text.length).toBeGreaterThan(0);
|
||||||
|
expect(response.text.split('<item>').length).toBe(3);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe('Test articles rendering', () => {
|
describe('Test articles rendering', () => {
|
||||||
test('404 article not found', (done) => {
|
test('404 article not found', (done) => {
|
||||||
request(app).get('/2019/05/06/untitled/').then((response) => {
|
request(app).get('/2019/05/06/untitled/').then((response) => {
|
||||||
|
|||||||
@@ -220,6 +220,8 @@ describe('Test article fetching', () => {
|
|||||||
const file = path.join(dir, testIndex);
|
const file = path.join(dir, testIndex);
|
||||||
utils.createEmptyDirs([dir]);
|
utils.createEmptyDirs([dir]);
|
||||||
utils.createEmptyFiles([file]);
|
utils.createEmptyFiles([file]);
|
||||||
|
const date = new Date(2019, 5, 5);
|
||||||
|
date.setUTCHours(0);
|
||||||
fw.fetchArticles((err, dict) => {
|
fw.fetchArticles((err, dict) => {
|
||||||
expect(err).toBeNull();
|
expect(err).toBeNull();
|
||||||
expect(dict).toBeDefined();
|
expect(dict).toBeDefined();
|
||||||
@@ -230,7 +232,7 @@ describe('Test article fetching', () => {
|
|||||||
year: 2019,
|
year: 2019,
|
||||||
month: 5,
|
month: 5,
|
||||||
day: 5,
|
day: 5,
|
||||||
date: new Date(2019, 5, 5),
|
date: date,
|
||||||
title: 'Untitled',
|
title: 'Untitled',
|
||||||
thumbnail: 'default.png',
|
thumbnail: 'default.png',
|
||||||
escapedTitle: 'untitled',
|
escapedTitle: 'untitled',
|
||||||
@@ -248,6 +250,8 @@ describe('Test article fetching', () => {
|
|||||||

|

|
||||||
this is some text
|
this is some text
|
||||||
`);
|
`);
|
||||||
|
const date = new Date(2019, 5, 5);
|
||||||
|
date.setUTCHours(0);
|
||||||
fw.fetchArticles((err, dict) => {
|
fw.fetchArticles((err, dict) => {
|
||||||
expect(err).toBeNull();
|
expect(err).toBeNull();
|
||||||
expect(dict).toBeDefined();
|
expect(dict).toBeDefined();
|
||||||
@@ -258,7 +262,7 @@ describe('Test article fetching', () => {
|
|||||||
year: 2019,
|
year: 2019,
|
||||||
month: 5,
|
month: 5,
|
||||||
day: 5,
|
day: 5,
|
||||||
date: new Date(2019, 5, 5),
|
date: date,
|
||||||
title: 'Title with : info !',
|
title: 'Title with : info !',
|
||||||
thumbnail: path.join('2019', '05', '05', './thumbnail.jpg'),
|
thumbnail: path.join('2019', '05', '05', './thumbnail.jpg'),
|
||||||
escapedTitle: 'title_with___info',
|
escapedTitle: 'title_with___info',
|
||||||
|
|||||||
@@ -1,100 +0,0 @@
|
|||||||
/* jshint -W117 */
|
|
||||||
const config = {
|
|
||||||
'loopback_url': 'http://test.test/',
|
|
||||||
'rss': {
|
|
||||||
'title': 'test rss',
|
|
||||||
'description': 'description',
|
|
||||||
'endpoint': '/rss',
|
|
||||||
'length': 2
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
const rss = require('../src/rss')(config);
|
|
||||||
|
|
||||||
test('empty rss', () => {
|
|
||||||
const xml = rss.get({});
|
|
||||||
expect(xml).toBe('<?xml version="1.0" encoding="UTF-8"?>' +
|
|
||||||
'<rss version="2.0">' +
|
|
||||||
'<title>test rss</title>' +
|
|
||||||
'<description>description</description>' +
|
|
||||||
'<link>http://test.test/</link>' +
|
|
||||||
'<lastBuildDate>' + new Date().toString() + '</lastBuildDate>' +
|
|
||||||
'<lastPubDate>' + new Date().toString() + '</lastPubDate>' +
|
|
||||||
'<ttl>60</ttl>' +
|
|
||||||
'</rss>');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('1 item', () => {
|
|
||||||
const data = {
|
|
||||||
'a': {
|
|
||||||
path: 'a',
|
|
||||||
realPath: 'b',
|
|
||||||
year: 2019,
|
|
||||||
month: 5,
|
|
||||||
day: 5,
|
|
||||||
date: new Date(2019, 5, 5),
|
|
||||||
title: 'Title with : info !',
|
|
||||||
thumbnail: '/2019/05/05/thumbnail.jpg/',
|
|
||||||
escapedTitle: 'title_with___info',
|
|
||||||
url: '/2019/05/05/title_with___info/',
|
|
||||||
}
|
|
||||||
};
|
|
||||||
const xml = rss.get(data);
|
|
||||||
expect(xml).toEqual('<?xml version="1.0" encoding="UTF-8"?>' +
|
|
||||||
'<rss version="2.0">' +
|
|
||||||
'<title>test rss</title>' +
|
|
||||||
'<description>description</description>' +
|
|
||||||
'<link>http://test.test/</link>' +
|
|
||||||
'<lastBuildDate>' + new Date().toString() + '</lastBuildDate>' +
|
|
||||||
'<lastPubDate>' + new Date().toString() + '</lastPubDate>' +
|
|
||||||
'<ttl>60</ttl>' +
|
|
||||||
'<item>' +
|
|
||||||
'<title>Title with : info !</title>' +
|
|
||||||
'<link>http://test.test/2019/05/05/title_with___info/</link>' +
|
|
||||||
'<pubDate>' + new Date(2019, 5, 5).toString() + '</pubDate>' +
|
|
||||||
'</item>' +
|
|
||||||
'</rss>');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('3 items only 2 shown sorted', () => {
|
|
||||||
const data = {
|
|
||||||
'a': {
|
|
||||||
path: '2019/05/05',
|
|
||||||
date: new Date(2019, 5, 5),
|
|
||||||
title: 'a',
|
|
||||||
url: 'a',
|
|
||||||
},
|
|
||||||
'b': {
|
|
||||||
path: '2018/05/05',
|
|
||||||
date: new Date(2018, 5, 5),
|
|
||||||
title: 'b',
|
|
||||||
url: 'b',
|
|
||||||
},
|
|
||||||
'c': {
|
|
||||||
path: '2020/05/05',
|
|
||||||
date: new Date(2020, 5, 5),
|
|
||||||
title: 'c',
|
|
||||||
url: 'c',
|
|
||||||
}
|
|
||||||
};
|
|
||||||
const xml = rss.get(data);
|
|
||||||
expect(xml).toEqual('<?xml version="1.0" encoding="UTF-8"?>' +
|
|
||||||
'<rss version="2.0">' +
|
|
||||||
'<title>test rss</title>' +
|
|
||||||
'<description>description</description>' +
|
|
||||||
'<link>http://test.test/</link>' +
|
|
||||||
'<lastBuildDate>' + new Date().toString() + '</lastBuildDate>' +
|
|
||||||
'<lastPubDate>' + new Date().toString() + '</lastPubDate>' +
|
|
||||||
'<ttl>60</ttl>' +
|
|
||||||
'<item>' +
|
|
||||||
'<title>c</title>' +
|
|
||||||
'<link>http://test.test/c</link>' +
|
|
||||||
'<pubDate>' + new Date(2020, 5, 5).toString() + '</pubDate>' +
|
|
||||||
'</item>' +
|
|
||||||
'<item>' +
|
|
||||||
'<title>a</title>' +
|
|
||||||
'<link>http://test.test/a</link>' +
|
|
||||||
'<pubDate>' + new Date(2019, 5, 5).toString() + '</pubDate>' +
|
|
||||||
'</item>' +
|
|
||||||
'</rss>');
|
|
||||||
});
|
|
||||||
Reference in New Issue
Block a user