mirror of
https://github.com/naturalcrit/homebrewery.git
synced 2026-01-03 12:42:41 +00:00
Merge branch 'master' into links-in-home-document
# Conflicts: # server.js
This commit is contained in:
@@ -17,6 +17,7 @@ const PAGE_HEIGHT = 1056;
|
|||||||
const PPR_THRESHOLD = 50;
|
const PPR_THRESHOLD = 50;
|
||||||
|
|
||||||
const BrewRenderer = createClass({
|
const BrewRenderer = createClass({
|
||||||
|
displayName : 'BrewRenderer',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
text : '',
|
text : '',
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ const _ = require('lodash');
|
|||||||
const cx = require('classnames');
|
const cx = require('classnames');
|
||||||
|
|
||||||
const ErrorBar = createClass({
|
const ErrorBar = createClass({
|
||||||
|
displayName : 'ErrorBar',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
errors : []
|
errors : []
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ const cx = require('classnames'); //Unused variable
|
|||||||
const DISMISS_KEY = 'dismiss_notification09-9-21';
|
const DISMISS_KEY = 'dismiss_notification09-9-21';
|
||||||
|
|
||||||
const NotificationPopup = createClass({
|
const NotificationPopup = createClass({
|
||||||
|
displayName : 'NotificationPopup',
|
||||||
getInitialState : function() {
|
getInitialState : function() {
|
||||||
return {
|
return {
|
||||||
notifications : {}
|
notifications : {}
|
||||||
|
|||||||
@@ -26,6 +26,7 @@ const splice = function(str, index, inject){
|
|||||||
|
|
||||||
|
|
||||||
const Editor = createClass({
|
const Editor = createClass({
|
||||||
|
displayName : 'Editor',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
brew : {
|
brew : {
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ const request = require('superagent');
|
|||||||
const SYSTEMS = ['5e', '4e', '3.5e', 'Pathfinder'];
|
const SYSTEMS = ['5e', '4e', '3.5e', 'Pathfinder'];
|
||||||
|
|
||||||
const MetadataEditor = createClass({
|
const MetadataEditor = createClass({
|
||||||
|
displayName : 'MetadataEditor',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
metadata : {
|
metadata : {
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ const execute = function(val, brew){
|
|||||||
};
|
};
|
||||||
|
|
||||||
const Snippetbar = createClass({
|
const Snippetbar = createClass({
|
||||||
|
displayName : 'SnippetBar',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
brew : {},
|
brew : {},
|
||||||
@@ -103,6 +104,7 @@ module.exports = Snippetbar;
|
|||||||
|
|
||||||
|
|
||||||
const SnippetGroup = createClass({
|
const SnippetGroup = createClass({
|
||||||
|
displayName : 'SnippetGroup',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
brew : {},
|
brew : {},
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ const NewPage = require('./pages/newPage/newPage.jsx');
|
|||||||
const PrintPage = require('./pages/printPage/printPage.jsx');
|
const PrintPage = require('./pages/printPage/printPage.jsx');
|
||||||
|
|
||||||
const Homebrew = createClass({
|
const Homebrew = createClass({
|
||||||
|
displayName : 'Homebrewery',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
url : '',
|
url : '',
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ const createClass = require('create-react-class');
|
|||||||
const Nav = require('naturalcrit/nav/nav.jsx');
|
const Nav = require('naturalcrit/nav/nav.jsx');
|
||||||
|
|
||||||
const Account = createClass({
|
const Account = createClass({
|
||||||
|
displayName : 'AccountNavItem',
|
||||||
getInitialState : function() {
|
getInitialState : function() {
|
||||||
return {
|
return {
|
||||||
url : ''
|
url : ''
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ const MAX_TITLE_LENGTH = 50;
|
|||||||
|
|
||||||
|
|
||||||
const EditTitle = createClass({
|
const EditTitle = createClass({
|
||||||
|
displayName : 'EditTitleNavItem',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
title : '',
|
title : '',
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ const Nav = require('naturalcrit/nav/nav.jsx');
|
|||||||
const PatreonNavItem = require('./patreon.navitem.jsx');
|
const PatreonNavItem = require('./patreon.navitem.jsx');
|
||||||
|
|
||||||
const Navbar = createClass({
|
const Navbar = createClass({
|
||||||
|
displayName : 'Navbar',
|
||||||
getInitialState : function() {
|
getInitialState : function() {
|
||||||
return {
|
return {
|
||||||
//showNonChromeWarning : false,
|
//showNonChromeWarning : false,
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ const VIEW_KEY = 'homebrewery-recently-viewed';
|
|||||||
|
|
||||||
|
|
||||||
const RecentItems = createClass({
|
const RecentItems = createClass({
|
||||||
|
DisplayName : 'RecentItems',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
storageKey : '',
|
storageKey : '',
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ const MAIN_URL = 'https://www.reddit.com/r/UnearthedArcana/submit?selftext=true'
|
|||||||
|
|
||||||
|
|
||||||
const RedditShare = createClass({
|
const RedditShare = createClass({
|
||||||
|
displayName : 'RedditShareNavItem',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
brew : {
|
brew : {
|
||||||
|
|||||||
@@ -27,6 +27,7 @@ const googleDriveInactive = require('../../googleDriveMono.png');
|
|||||||
const SAVE_TIMEOUT = 3000;
|
const SAVE_TIMEOUT = 3000;
|
||||||
|
|
||||||
const EditPage = createClass({
|
const EditPage = createClass({
|
||||||
|
displayName : 'EditPage',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
brew : {
|
brew : {
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ const BrewRenderer = require('../../brewRenderer/brewRenderer.jsx');
|
|||||||
|
|
||||||
|
|
||||||
const HomePage = createClass({
|
const HomePage = createClass({
|
||||||
|
displayName : 'HomePage',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
brew : {
|
brew : {
|
||||||
|
|||||||
@@ -23,6 +23,7 @@ const METAKEY = 'homebrewery-new-meta';
|
|||||||
|
|
||||||
|
|
||||||
const NewPage = createClass({
|
const NewPage = createClass({
|
||||||
|
displayName : 'NewPage',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
brew : {
|
brew : {
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ const STYLEKEY = 'homebrewery-new-style';
|
|||||||
const METAKEY = 'homebrewery-new-meta';
|
const METAKEY = 'homebrewery-new-meta';
|
||||||
|
|
||||||
const PrintPage = createClass({
|
const PrintPage = createClass({
|
||||||
|
displayName : 'PrintPage',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
query : {},
|
query : {},
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ const BrewRenderer = require('../../brewRenderer/brewRenderer.jsx');
|
|||||||
|
|
||||||
|
|
||||||
const SharePage = createClass({
|
const SharePage = createClass({
|
||||||
|
displayName : 'SharePage',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
brew : {
|
brew : {
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ const googleDriveIcon = require('../../../googleDrive.png');
|
|||||||
const dedent = require('dedent-tabs').default;
|
const dedent = require('dedent-tabs').default;
|
||||||
|
|
||||||
const BrewItem = createClass({
|
const BrewItem = createClass({
|
||||||
|
displayName : 'BrewItem',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
brew : {
|
brew : {
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ const ReportIssue = require('../../navbar/issue.navitem.jsx');
|
|||||||
|
|
||||||
|
|
||||||
const UserPage = createClass({
|
const UserPage = createClass({
|
||||||
|
displayName : 'UserPage',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
username : '',
|
username : '',
|
||||||
|
|||||||
8236
package-lock.json
generated
8236
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
33
package.json
33
package.json
@@ -18,8 +18,8 @@
|
|||||||
"lint:dry": "eslint **/*.{js,jsx}",
|
"lint:dry": "eslint **/*.{js,jsx}",
|
||||||
"circleci": "npm test && eslint **/*.{js,jsx} --max-warnings=0",
|
"circleci": "npm test && eslint **/*.{js,jsx} --max-warnings=0",
|
||||||
"verify": "npm run lint && npm test",
|
"verify": "npm run lint && npm test",
|
||||||
"test": "pico-check",
|
"test": "jest",
|
||||||
"test:dev": "pico-check -v -w",
|
"test:dev": "jest --verbose --watch",
|
||||||
"phb": "node scripts/phb.js",
|
"phb": "node scripts/phb.js",
|
||||||
"prod": "set NODE_ENV=production && npm run build",
|
"prod": "set NODE_ENV=production && npm run build",
|
||||||
"postinstall": "npm run buildall",
|
"postinstall": "npm run buildall",
|
||||||
@@ -30,8 +30,11 @@
|
|||||||
"eslintIgnore": [
|
"eslintIgnore": [
|
||||||
"build/*"
|
"build/*"
|
||||||
],
|
],
|
||||||
"pico-check": {
|
"jest": {
|
||||||
"require": "./tests/test.init.js"
|
"modulePaths": [
|
||||||
|
"mode_modules",
|
||||||
|
"shared"
|
||||||
|
]
|
||||||
},
|
},
|
||||||
"babel": {
|
"babel": {
|
||||||
"presets": [
|
"presets": [
|
||||||
@@ -40,13 +43,13 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@babel/core": "^7.16.5",
|
"@babel/core": "^7.16.12",
|
||||||
"@babel/plugin-transform-runtime": "^7.16.5",
|
"@babel/plugin-transform-runtime": "^7.16.10",
|
||||||
"@babel/preset-env": "^7.16.5",
|
"@babel/preset-env": "^7.16.11",
|
||||||
"@babel/preset-react": "^7.16.5",
|
"@babel/preset-react": "^7.16.7",
|
||||||
"body-parser": "^1.19.1",
|
"body-parser": "^1.19.1",
|
||||||
"classnames": "^2.3.1",
|
"classnames": "^2.3.1",
|
||||||
"codemirror": "^5.65.0",
|
"codemirror": "^5.65.1",
|
||||||
"cookie-parser": "^1.4.6",
|
"cookie-parser": "^1.4.6",
|
||||||
"create-react-class": "^15.7.0",
|
"create-react-class": "^15.7.0",
|
||||||
"dedent-tabs": "^0.10.1",
|
"dedent-tabs": "^0.10.1",
|
||||||
@@ -59,15 +62,15 @@
|
|||||||
"jwt-simple": "^0.5.6",
|
"jwt-simple": "^0.5.6",
|
||||||
"less": "^3.13.1",
|
"less": "^3.13.1",
|
||||||
"lodash": "^4.17.21",
|
"lodash": "^4.17.21",
|
||||||
"marked": "4.0.8",
|
"marked": "4.0.10",
|
||||||
"marked-extended-tables": "^1.0.3",
|
"marked-extended-tables": "^1.0.3",
|
||||||
"markedLegacy": "npm:marked@^0.3.19",
|
"markedLegacy": "npm:marked@^0.3.19",
|
||||||
"moment": "^2.29.1",
|
"moment": "^2.29.1",
|
||||||
"mongoose": "^6.1.3",
|
"mongoose": "^6.1.7",
|
||||||
"nanoid": "3.1.30",
|
"nanoid": "3.2.0",
|
||||||
"nconf": "^0.11.3",
|
"nconf": "^0.11.3",
|
||||||
"prop-types": "15.8.0",
|
"prop-types": "15.8.0",
|
||||||
"query-string": "7.0.1",
|
"query-string": "7.1.0",
|
||||||
"react": "^16.14.0",
|
"react": "^16.14.0",
|
||||||
"react-dom": "^16.14.0",
|
"react-dom": "^16.14.0",
|
||||||
"react-frame-component": "4.1.3",
|
"react-frame-component": "4.1.3",
|
||||||
@@ -77,8 +80,8 @@
|
|||||||
"vitreum": "git+https://git@github.com/calculuschild/vitreum.git"
|
"vitreum": "git+https://git@github.com/calculuschild/vitreum.git"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"eslint": "^8.5.0",
|
"eslint": "^8.7.0",
|
||||||
"eslint-plugin-react": "^7.28.0",
|
"eslint-plugin-react": "^7.28.0",
|
||||||
"pico-check": "^2.2.0"
|
"jest": "^27.4.5"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
309
server.js
309
server.js
@@ -1,75 +1,5 @@
|
|||||||
/*eslint max-lines: ["warn", {"max": 300, "skipBlankLines": true, "skipComments": true}]*/
|
const DB = require('./server/db.js');
|
||||||
const _ = require('lodash');
|
const server = require('./server/app.js');
|
||||||
const jwt = require('jwt-simple');
|
|
||||||
const express = require('express');
|
|
||||||
const yaml = require('js-yaml');
|
|
||||||
const app = express();
|
|
||||||
|
|
||||||
const homebrewApi = require('./server/homebrew.api.js');
|
|
||||||
const GoogleActions = require('./server/googleActions.js');
|
|
||||||
const serveCompressedStaticAssets = require('./server/static-assets.mv.js');
|
|
||||||
const sanitizeFilename = require('sanitize-filename');
|
|
||||||
const asyncHandler = require('express-async-handler');
|
|
||||||
|
|
||||||
const brewAccessTypes = ['edit', 'share', 'raw'];
|
|
||||||
|
|
||||||
//Get the brew object from the HB database or Google Drive
|
|
||||||
const getBrewFromId = asyncHandler(async (id, accessType)=>{
|
|
||||||
if(!brewAccessTypes.includes(accessType))
|
|
||||||
throw ('Invalid Access Type when getting brew');
|
|
||||||
let brew;
|
|
||||||
if(id.length > 12) {
|
|
||||||
const googleId = id.slice(0, -12);
|
|
||||||
id = id.slice(-12);
|
|
||||||
brew = await GoogleActions.readFileMetadata(config.get('google_api_key'), googleId, id, accessType);
|
|
||||||
} else {
|
|
||||||
brew = await HomebrewModel.get(accessType == 'edit' ? { editId: id } : { shareId: id });
|
|
||||||
brew = brew.toObject(); // Convert MongoDB object to standard Javascript Object
|
|
||||||
}
|
|
||||||
|
|
||||||
brew = sanitizeBrew(brew, accessType === 'edit' ? false : true);
|
|
||||||
//Split brew.text into text and style
|
|
||||||
//unless the Access Type is RAW, in which case return immediately
|
|
||||||
if(accessType == 'raw') {
|
|
||||||
return brew;
|
|
||||||
}
|
|
||||||
splitTextStyleAndMetadata(brew);
|
|
||||||
return brew;
|
|
||||||
});
|
|
||||||
|
|
||||||
const sanitizeBrew = (brew, full=false)=>{
|
|
||||||
delete brew._id;
|
|
||||||
delete brew.__v;
|
|
||||||
if(full){
|
|
||||||
delete brew.editId;
|
|
||||||
}
|
|
||||||
return brew;
|
|
||||||
};
|
|
||||||
|
|
||||||
const splitTextStyleAndMetadata = (brew)=>{
|
|
||||||
brew.text = brew.text.replaceAll('\r\n', '\n');
|
|
||||||
if(brew.text.startsWith('```metadata')) {
|
|
||||||
const index = brew.text.indexOf('```\n\n');
|
|
||||||
const metadataSection = brew.text.slice(12, index - 1);
|
|
||||||
const metadata = yaml.load(metadataSection);
|
|
||||||
Object.assign(brew, _.pick(metadata, ['title', 'description', 'tags', 'systems', 'renderer']));
|
|
||||||
brew.text = brew.text.slice(index + 5);
|
|
||||||
}
|
|
||||||
if(brew.text.startsWith('```css')) {
|
|
||||||
const index = brew.text.indexOf('```\n\n');
|
|
||||||
brew.style = brew.text.slice(7, index - 1);
|
|
||||||
brew.text = brew.text.slice(index + 5);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
app.use('/', serveCompressedStaticAssets(`${__dirname}/build`));
|
|
||||||
|
|
||||||
process.chdir(__dirname);
|
|
||||||
|
|
||||||
//app.use(express.static(`${__dirname}/build`));
|
|
||||||
app.use(require('body-parser').json({ limit: '25mb' }));
|
|
||||||
app.use(require('cookie-parser')());
|
|
||||||
app.use(require('./server/forcessl.mw.js'));
|
|
||||||
|
|
||||||
const config = require('nconf')
|
const config = require('nconf')
|
||||||
.argv()
|
.argv()
|
||||||
@@ -77,234 +7,11 @@ const config = require('nconf')
|
|||||||
.file('environment', { file: `config/${process.env.NODE_ENV}.json` })
|
.file('environment', { file: `config/${process.env.NODE_ENV}.json` })
|
||||||
.file('defaults', { file: 'config/default.json' });
|
.file('defaults', { file: 'config/default.json' });
|
||||||
|
|
||||||
//DB
|
DB.connect(config).then(()=>{
|
||||||
const mongoose = require('mongoose');
|
// Ensure that we have successfully connected to the database
|
||||||
mongoose.connect(config.get('mongodb_uri') || config.get('mongolab_uri') || 'mongodb://localhost/naturalcrit',
|
// before launching server
|
||||||
{ retryWrites: false });
|
const PORT = process.env.PORT || config.get('web_port') || 8000;
|
||||||
mongoose.connection.on('error', (err)=>{
|
server.app.listen(PORT, ()=>{
|
||||||
console.log('Error : Could not connect to a Mongo Database.');
|
console.log(`server on port: ${PORT}`);
|
||||||
console.log(' If you are running locally, make sure mongodb.exe is running.');
|
|
||||||
console.log(err);
|
|
||||||
throw 'Can not connect to Mongo';
|
|
||||||
});
|
|
||||||
|
|
||||||
//Account Middleware
|
|
||||||
app.use((req, res, next)=>{
|
|
||||||
if(req.cookies && req.cookies.nc_session){
|
|
||||||
try {
|
|
||||||
req.account = jwt.decode(req.cookies.nc_session, config.get('secret'));
|
|
||||||
//console.log("Just loaded up JWT from cookie:");
|
|
||||||
//console.log(req.account);
|
|
||||||
} catch (e){}
|
|
||||||
}
|
|
||||||
|
|
||||||
req.config = {
|
|
||||||
google_client_id : config.get('google_client_id'),
|
|
||||||
google_client_secret : config.get('google_client_secret')
|
|
||||||
};
|
|
||||||
return next();
|
|
||||||
});
|
|
||||||
|
|
||||||
app.use(homebrewApi);
|
|
||||||
app.use(require('./server/admin.api.js'));
|
|
||||||
|
|
||||||
const HomebrewModel = require('./server/homebrew.model.js').model;
|
|
||||||
const welcomeText = require('fs').readFileSync('./client/homebrew/pages/homePage/welcome_msg.md', 'utf8');
|
|
||||||
const welcomeTextV3 = require('fs').readFileSync('./client/homebrew/pages/homePage/welcome_msg_v3.md', 'utf8');
|
|
||||||
const changelogText = require('fs').readFileSync('./changelog.md', 'utf8');
|
|
||||||
const faqText = require('fs').readFileSync('./faq.md', 'utf8');
|
|
||||||
|
|
||||||
String.prototype.replaceAll = function(s, r){return this.split(s).join(r);};
|
|
||||||
|
|
||||||
//Robots.txt
|
|
||||||
app.get('/robots.txt', (req, res)=>{
|
|
||||||
return res.sendFile(`${__dirname}/robots.txt`);
|
|
||||||
});
|
|
||||||
|
|
||||||
//Home page
|
|
||||||
app.get('/', async (req, res, next)=>{
|
|
||||||
const brew = {
|
|
||||||
text : welcomeText
|
|
||||||
};
|
|
||||||
splitTextStyleAndMetadata(brew);
|
|
||||||
req.brew = brew;
|
|
||||||
return next();
|
|
||||||
});
|
|
||||||
|
|
||||||
//Home page v3
|
|
||||||
app.get('/v3_preview', async (req, res, next)=>{
|
|
||||||
const brew = {
|
|
||||||
text : welcomeTextV3,
|
|
||||||
renderer : 'V3'
|
|
||||||
};
|
|
||||||
splitTextStyleAndMetadata(brew);
|
|
||||||
req.brew = brew;
|
|
||||||
return next();
|
|
||||||
});
|
|
||||||
|
|
||||||
//Changelog page
|
|
||||||
app.get('/changelog', async (req, res, next)=>{
|
|
||||||
const brew = {
|
|
||||||
title : 'Changelog',
|
|
||||||
text : changelogText,
|
|
||||||
renderer : 'V3'
|
|
||||||
};
|
|
||||||
splitTextStyleAndMetadata(brew);
|
|
||||||
req.brew = brew;
|
|
||||||
return next();
|
|
||||||
});
|
|
||||||
|
|
||||||
//FAQ page
|
|
||||||
app.get('/faq', async (req, res, next)=>{
|
|
||||||
const brew = {
|
|
||||||
title : 'FAQ',
|
|
||||||
text : faqText,
|
|
||||||
renderer : 'V3'
|
|
||||||
};
|
|
||||||
splitTextStyleAndMetadata(brew);
|
|
||||||
req.brew = brew;
|
|
||||||
return next();
|
|
||||||
});
|
|
||||||
|
|
||||||
//Source page
|
|
||||||
app.get('/source/:id', asyncHandler(async (req, res)=>{
|
|
||||||
const brew = await getBrewFromId(req.params.id, 'raw');
|
|
||||||
|
|
||||||
const replaceStrings = { '&': '&', '<': '<', '>': '>' };
|
|
||||||
let text = brew.text;
|
|
||||||
for (const replaceStr in replaceStrings) {
|
|
||||||
text = text.replaceAll(replaceStr, replaceStrings[replaceStr]);
|
|
||||||
}
|
|
||||||
text = `<code><pre style="white-space: pre-wrap;">${text}</pre></code>`;
|
|
||||||
res.status(200).send(text);
|
|
||||||
}));
|
|
||||||
|
|
||||||
//Download brew source page
|
|
||||||
app.get('/download/:id', asyncHandler(async (req, res)=>{
|
|
||||||
const brew = await getBrewFromId(req.params.id, 'raw');
|
|
||||||
const prefix = 'HB - ';
|
|
||||||
|
|
||||||
let fileName = sanitizeFilename(`${prefix}${brew.title}`).replaceAll(' ', '');
|
|
||||||
if(!fileName || !fileName.length) { fileName = `${prefix}-Untitled-Brew`; };
|
|
||||||
res.set({
|
|
||||||
'Cache-Control' : 'no-cache',
|
|
||||||
'Content-Type' : 'text/plain',
|
|
||||||
'Content-Disposition' : `attachment; filename="${fileName}.txt"`
|
|
||||||
});
|
});
|
||||||
res.status(200).send(brew.text);
|
|
||||||
}));
|
|
||||||
|
|
||||||
//User Page
|
|
||||||
app.get('/user/:username', async (req, res, next)=>{
|
|
||||||
const ownAccount = req.account && (req.account.username == req.params.username);
|
|
||||||
|
|
||||||
let brews = await HomebrewModel.getByUser(req.params.username, ownAccount)
|
|
||||||
.catch((err)=>{
|
|
||||||
console.log(err);
|
|
||||||
});
|
|
||||||
|
|
||||||
if(ownAccount && req?.account?.googleId){
|
|
||||||
const googleBrews = await GoogleActions.listGoogleBrews(req, res)
|
|
||||||
.catch((err)=>{
|
|
||||||
console.error(err);
|
|
||||||
});
|
|
||||||
|
|
||||||
if(googleBrews)
|
|
||||||
brews = _.concat(brews, googleBrews);
|
|
||||||
}
|
|
||||||
|
|
||||||
req.brews = _.map(brews, (brew)=>{
|
|
||||||
return sanitizeBrew(brew, !ownAccount);
|
|
||||||
});
|
|
||||||
|
|
||||||
return next();
|
|
||||||
});
|
|
||||||
|
|
||||||
//Edit Page
|
|
||||||
app.get('/edit/:id', asyncHandler(async (req, res, next)=>{
|
|
||||||
res.header('Cache-Control', 'no-cache, no-store'); //reload the latest saved brew when pressing back button, not the cached version before save.
|
|
||||||
const brew = await getBrewFromId(req.params.id, 'edit');
|
|
||||||
req.brew = brew;
|
|
||||||
return next();
|
|
||||||
}));
|
|
||||||
|
|
||||||
//New Page
|
|
||||||
app.get('/new/:id', asyncHandler(async (req, res, next)=>{
|
|
||||||
const brew = await getBrewFromId(req.params.id, 'share');
|
|
||||||
brew.title = `CLONE - ${brew.title}`;
|
|
||||||
req.brew = brew;
|
|
||||||
return next();
|
|
||||||
}));
|
|
||||||
|
|
||||||
//Share Page
|
|
||||||
app.get('/share/:id', asyncHandler(async (req, res, next)=>{
|
|
||||||
const brew = await getBrewFromId(req.params.id, 'share');
|
|
||||||
|
|
||||||
if(req.params.id.length > 12) {
|
|
||||||
const googleId = req.params.id.slice(0, -12);
|
|
||||||
const shareId = req.params.id.slice(-12);
|
|
||||||
await GoogleActions.increaseView(googleId, shareId, 'share', brew)
|
|
||||||
.catch((err)=>{next(err);});
|
|
||||||
} else {
|
|
||||||
await HomebrewModel.increaseView({ shareId: brew.shareId });
|
|
||||||
}
|
|
||||||
|
|
||||||
req.brew = brew;
|
|
||||||
return next();
|
|
||||||
}));
|
|
||||||
|
|
||||||
//Print Page
|
|
||||||
app.get('/print/:id', asyncHandler(async (req, res, next)=>{
|
|
||||||
const brew = await getBrewFromId(req.params.id, 'share');
|
|
||||||
req.brew = brew;
|
|
||||||
return next();
|
|
||||||
}));
|
|
||||||
|
|
||||||
//Render the page
|
|
||||||
const templateFn = require('./client/template.js');
|
|
||||||
app.use((req, res)=>{
|
|
||||||
const props = {
|
|
||||||
version : require('./package.json').version,
|
|
||||||
url : req.originalUrl,
|
|
||||||
brew : req.brew,
|
|
||||||
brews : req.brews,
|
|
||||||
googleBrews : req.googleBrews,
|
|
||||||
account : req.account,
|
|
||||||
enable_v3 : config.get('enable_v3')
|
|
||||||
};
|
|
||||||
templateFn('homebrew', title = req.brew ? req.brew.title : '', props)
|
|
||||||
.then((page)=>{ res.send(page); })
|
|
||||||
.catch((err)=>{
|
|
||||||
console.log(err);
|
|
||||||
return res.sendStatus(500);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
//v=====----- Error-Handling Middleware -----=====v//
|
|
||||||
//Format Errors so all fields will be sent
|
|
||||||
const replaceErrors = (key, value)=>{
|
|
||||||
if(value instanceof Error) {
|
|
||||||
const error = {};
|
|
||||||
Object.getOwnPropertyNames(value).forEach(function (key) {
|
|
||||||
error[key] = value[key];
|
|
||||||
});
|
|
||||||
return error;
|
|
||||||
}
|
|
||||||
return value;
|
|
||||||
};
|
|
||||||
|
|
||||||
const getPureError = (error)=>{
|
|
||||||
return JSON.parse(JSON.stringify(error, replaceErrors));
|
|
||||||
};
|
|
||||||
|
|
||||||
app.use((err, req, res, next)=>{
|
|
||||||
const status = err.status || 500;
|
|
||||||
console.error(err);
|
|
||||||
res.status(status).send(getPureError(err));
|
|
||||||
});
|
|
||||||
//^=====--------------------------------------=====^//
|
|
||||||
|
|
||||||
const PORT = process.env.PORT || config.get('web_port') || 8000;
|
|
||||||
app.listen(PORT, ()=>{
|
|
||||||
console.log(`server on port:${PORT}`);
|
|
||||||
});
|
});
|
||||||
|
|||||||
298
server/app.js
Normal file
298
server/app.js
Normal file
@@ -0,0 +1,298 @@
|
|||||||
|
/*eslint max-lines: ["warn", {"max": 300, "skipBlankLines": true, "skipComments": true}]*/
|
||||||
|
const _ = require('lodash');
|
||||||
|
const jwt = require('jwt-simple');
|
||||||
|
const express = require('express');
|
||||||
|
const yaml = require('js-yaml');
|
||||||
|
const app = express();
|
||||||
|
|
||||||
|
const homebrewApi = require('./homebrew.api.js');
|
||||||
|
const GoogleActions = require('./googleActions.js');
|
||||||
|
const serveCompressedStaticAssets = require('./static-assets.mv.js');
|
||||||
|
const sanitizeFilename = require('sanitize-filename');
|
||||||
|
const asyncHandler = require('express-async-handler');
|
||||||
|
|
||||||
|
const brewAccessTypes = ['edit', 'share', 'raw'];
|
||||||
|
|
||||||
|
//Get the brew object from the HB database or Google Drive
|
||||||
|
const getBrewFromId = asyncHandler(async (id, accessType)=>{
|
||||||
|
if(!brewAccessTypes.includes(accessType))
|
||||||
|
throw ('Invalid Access Type when getting brew');
|
||||||
|
let brew;
|
||||||
|
if(id.length > 12) {
|
||||||
|
const googleId = id.slice(0, -12);
|
||||||
|
id = id.slice(-12);
|
||||||
|
brew = await GoogleActions.readFileMetadata(config.get('google_api_key'), googleId, id, accessType);
|
||||||
|
} else {
|
||||||
|
brew = await HomebrewModel.get(accessType == 'edit' ? { editId: id } : { shareId: id });
|
||||||
|
brew = brew.toObject(); // Convert MongoDB object to standard Javascript Object
|
||||||
|
}
|
||||||
|
|
||||||
|
brew = sanitizeBrew(brew, accessType === 'edit' ? false : true);
|
||||||
|
//Split brew.text into text and style
|
||||||
|
//unless the Access Type is RAW, in which case return immediately
|
||||||
|
if(accessType == 'raw') {
|
||||||
|
return brew;
|
||||||
|
}
|
||||||
|
splitTextStyleAndMetadata(brew);
|
||||||
|
return brew;
|
||||||
|
});
|
||||||
|
|
||||||
|
const sanitizeBrew = (brew, full=false)=>{
|
||||||
|
delete brew._id;
|
||||||
|
delete brew.__v;
|
||||||
|
if(full){
|
||||||
|
delete brew.editId;
|
||||||
|
}
|
||||||
|
return brew;
|
||||||
|
};
|
||||||
|
|
||||||
|
const splitTextStyleAndMetadata = (brew)=>{
|
||||||
|
brew.text = brew.text.replaceAll('\r\n', '\n');
|
||||||
|
if(brew.text.startsWith('```metadata')) {
|
||||||
|
const index = brew.text.indexOf('```\n\n');
|
||||||
|
const metadataSection = brew.text.slice(12, index - 1);
|
||||||
|
const metadata = yaml.load(metadataSection);
|
||||||
|
Object.assign(brew, _.pick(metadata, ['title', 'description', 'tags', 'systems', 'renderer']));
|
||||||
|
brew.text = brew.text.slice(index + 5);
|
||||||
|
}
|
||||||
|
if(brew.text.startsWith('```css')) {
|
||||||
|
const index = brew.text.indexOf('```\n\n');
|
||||||
|
brew.style = brew.text.slice(7, index - 1);
|
||||||
|
brew.text = brew.text.slice(index + 5);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
app.use('/', serveCompressedStaticAssets(`${__dirname}/../build`));
|
||||||
|
|
||||||
|
process.chdir(__dirname);
|
||||||
|
|
||||||
|
//app.use(express.static(`${__dirname}/build`));
|
||||||
|
app.use(require('body-parser').json({ limit: '25mb' }));
|
||||||
|
app.use(require('cookie-parser')());
|
||||||
|
app.use(require('./forcessl.mw.js'));
|
||||||
|
|
||||||
|
// FIXME: the config should be passed as an argument for the app
|
||||||
|
const config = require('nconf')
|
||||||
|
.argv()
|
||||||
|
.env({ lowerCase: true })
|
||||||
|
.file('environment', { file: `config/${process.env.NODE_ENV}.json` })
|
||||||
|
.file('defaults', { file: 'config/default.json' });
|
||||||
|
|
||||||
|
//Account Middleware
|
||||||
|
app.use((req, res, next)=>{
|
||||||
|
if(req.cookies && req.cookies.nc_session){
|
||||||
|
try {
|
||||||
|
req.account = jwt.decode(req.cookies.nc_session, config.get('secret'));
|
||||||
|
//console.log("Just loaded up JWT from cookie:");
|
||||||
|
//console.log(req.account);
|
||||||
|
} catch (e){}
|
||||||
|
}
|
||||||
|
|
||||||
|
req.config = {
|
||||||
|
google_client_id : config.get('google_client_id'),
|
||||||
|
google_client_secret : config.get('google_client_secret')
|
||||||
|
};
|
||||||
|
return next();
|
||||||
|
});
|
||||||
|
|
||||||
|
app.use(homebrewApi);
|
||||||
|
app.use(require('./admin.api.js'));
|
||||||
|
|
||||||
|
const HomebrewModel = require('./homebrew.model.js').model;
|
||||||
|
const welcomeText = require('fs').readFileSync('./../client/homebrew/pages/homePage/welcome_msg.md', 'utf8');
|
||||||
|
const welcomeTextV3 = require('fs').readFileSync('./../client/homebrew/pages/homePage/welcome_msg_v3.md', 'utf8');
|
||||||
|
const changelogText = require('fs').readFileSync('./../changelog.md', 'utf8');
|
||||||
|
const faqText = require('fs').readFileSync('./../faq.md', 'utf8');
|
||||||
|
|
||||||
|
String.prototype.replaceAll = function(s, r){return this.split(s).join(r);};
|
||||||
|
|
||||||
|
//Robots.txt
|
||||||
|
app.get('/robots.txt', (req, res)=>{
|
||||||
|
return res.sendFile(`${__dirname}/robots.txt`);
|
||||||
|
});
|
||||||
|
|
||||||
|
//Home page
|
||||||
|
app.get('/', async (req, res, next)=>{
|
||||||
|
const brew = {
|
||||||
|
text : welcomeText
|
||||||
|
};
|
||||||
|
req.brew = brew;
|
||||||
|
return next();
|
||||||
|
});
|
||||||
|
|
||||||
|
//Home page v3
|
||||||
|
app.get('/v3_preview', async (req, res, next)=>{
|
||||||
|
const brew = {
|
||||||
|
text : welcomeTextV3,
|
||||||
|
renderer : 'V3'
|
||||||
|
};
|
||||||
|
splitTextStyleAndMetadata(brew);
|
||||||
|
req.brew = brew;
|
||||||
|
return next();
|
||||||
|
});
|
||||||
|
|
||||||
|
//Changelog page
|
||||||
|
app.get('/changelog', async (req, res, next)=>{
|
||||||
|
const brew = {
|
||||||
|
title : 'Changelog',
|
||||||
|
text : changelogText,
|
||||||
|
renderer : 'V3'
|
||||||
|
};
|
||||||
|
splitTextStyleAndMetadata(brew);
|
||||||
|
req.brew = brew;
|
||||||
|
return next();
|
||||||
|
});
|
||||||
|
|
||||||
|
//FAQ page
|
||||||
|
app.get('/faq', async (req, res, next)=>{
|
||||||
|
const brew = {
|
||||||
|
title : 'FAQ',
|
||||||
|
text : faqText,
|
||||||
|
renderer : 'V3'
|
||||||
|
};
|
||||||
|
splitTextStyleAndMetadata(brew);
|
||||||
|
req.brew = brew;
|
||||||
|
return next();
|
||||||
|
});
|
||||||
|
|
||||||
|
//Source page
|
||||||
|
app.get('/source/:id', asyncHandler(async (req, res)=>{
|
||||||
|
const brew = await getBrewFromId(req.params.id, 'raw');
|
||||||
|
|
||||||
|
const replaceStrings = { '&': '&', '<': '<', '>': '>' };
|
||||||
|
let text = brew.text;
|
||||||
|
for (const replaceStr in replaceStrings) {
|
||||||
|
text = text.replaceAll(replaceStr, replaceStrings[replaceStr]);
|
||||||
|
}
|
||||||
|
text = `<code><pre style="white-space: pre-wrap;">${text}</pre></code>`;
|
||||||
|
res.status(200).send(text);
|
||||||
|
}));
|
||||||
|
|
||||||
|
//Download brew source page
|
||||||
|
app.get('/download/:id', asyncHandler(async (req, res)=>{
|
||||||
|
const brew = await getBrewFromId(req.params.id, 'raw');
|
||||||
|
const prefix = 'HB - ';
|
||||||
|
|
||||||
|
let fileName = sanitizeFilename(`${prefix}${brew.title}`).replaceAll(' ', '');
|
||||||
|
if(!fileName || !fileName.length) { fileName = `${prefix}-Untitled-Brew`; };
|
||||||
|
res.set({
|
||||||
|
'Cache-Control' : 'no-cache',
|
||||||
|
'Content-Type' : 'text/plain',
|
||||||
|
'Content-Disposition' : `attachment; filename="${fileName}.txt"`
|
||||||
|
});
|
||||||
|
res.status(200).send(brew.text);
|
||||||
|
}));
|
||||||
|
|
||||||
|
//User Page
|
||||||
|
app.get('/user/:username', async (req, res, next)=>{
|
||||||
|
const ownAccount = req.account && (req.account.username == req.params.username);
|
||||||
|
|
||||||
|
let brews = await HomebrewModel.getByUser(req.params.username, ownAccount)
|
||||||
|
.catch((err)=>{
|
||||||
|
console.log(err);
|
||||||
|
});
|
||||||
|
|
||||||
|
if(ownAccount && req?.account?.googleId){
|
||||||
|
const googleBrews = await GoogleActions.listGoogleBrews(req, res)
|
||||||
|
.catch((err)=>{
|
||||||
|
console.error(err);
|
||||||
|
});
|
||||||
|
|
||||||
|
if(googleBrews)
|
||||||
|
brews = _.concat(brews, googleBrews);
|
||||||
|
}
|
||||||
|
|
||||||
|
req.brews = _.map(brews, (brew)=>{
|
||||||
|
return sanitizeBrew(brew, !ownAccount);
|
||||||
|
});
|
||||||
|
|
||||||
|
return next();
|
||||||
|
});
|
||||||
|
|
||||||
|
//Edit Page
|
||||||
|
app.get('/edit/:id', asyncHandler(async (req, res, next)=>{
|
||||||
|
res.header('Cache-Control', 'no-cache, no-store'); //reload the latest saved brew when pressing back button, not the cached version before save.
|
||||||
|
const brew = await getBrewFromId(req.params.id, 'edit');
|
||||||
|
req.brew = brew;
|
||||||
|
return next();
|
||||||
|
}));
|
||||||
|
|
||||||
|
//New Page
|
||||||
|
app.get('/new/:id', asyncHandler(async (req, res, next)=>{
|
||||||
|
const brew = await getBrewFromId(req.params.id, 'share');
|
||||||
|
brew.title = `CLONE - ${brew.title}`;
|
||||||
|
req.brew = brew;
|
||||||
|
return next();
|
||||||
|
}));
|
||||||
|
|
||||||
|
//Share Page
|
||||||
|
app.get('/share/:id', asyncHandler(async (req, res, next)=>{
|
||||||
|
const brew = await getBrewFromId(req.params.id, 'share');
|
||||||
|
|
||||||
|
if(req.params.id.length > 12) {
|
||||||
|
const googleId = req.params.id.slice(0, -12);
|
||||||
|
const shareId = req.params.id.slice(-12);
|
||||||
|
await GoogleActions.increaseView(googleId, shareId, 'share', brew)
|
||||||
|
.catch((err)=>{next(err);});
|
||||||
|
} else {
|
||||||
|
await HomebrewModel.increaseView({ shareId: brew.shareId });
|
||||||
|
}
|
||||||
|
|
||||||
|
req.brew = brew;
|
||||||
|
return next();
|
||||||
|
}));
|
||||||
|
|
||||||
|
//Print Page
|
||||||
|
app.get('/print/:id', asyncHandler(async (req, res, next)=>{
|
||||||
|
const brew = await getBrewFromId(req.params.id, 'share');
|
||||||
|
req.brew = brew;
|
||||||
|
return next();
|
||||||
|
}));
|
||||||
|
|
||||||
|
//Render the page
|
||||||
|
const templateFn = require('./../client/template.js');
|
||||||
|
app.use((req, res)=>{
|
||||||
|
const props = {
|
||||||
|
version : require('./../package.json').version,
|
||||||
|
url : req.originalUrl,
|
||||||
|
brew : req.brew,
|
||||||
|
brews : req.brews,
|
||||||
|
googleBrews : req.googleBrews,
|
||||||
|
account : req.account,
|
||||||
|
enable_v3 : config.get('enable_v3')
|
||||||
|
};
|
||||||
|
templateFn('homebrew', title = req.brew ? req.brew.title : '', props)
|
||||||
|
.then((page)=>{ res.send(page); })
|
||||||
|
.catch((err)=>{
|
||||||
|
console.log(err);
|
||||||
|
return res.sendStatus(500);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
//v=====----- Error-Handling Middleware -----=====v//
|
||||||
|
//Format Errors so all fields will be sent
|
||||||
|
const replaceErrors = (key, value)=>{
|
||||||
|
if(value instanceof Error) {
|
||||||
|
const error = {};
|
||||||
|
Object.getOwnPropertyNames(value).forEach(function (key) {
|
||||||
|
error[key] = value[key];
|
||||||
|
});
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getPureError = (error)=>{
|
||||||
|
return JSON.parse(JSON.stringify(error, replaceErrors));
|
||||||
|
};
|
||||||
|
|
||||||
|
app.use((err, req, res, next)=>{
|
||||||
|
const status = err.status || 500;
|
||||||
|
console.error(err);
|
||||||
|
res.status(status).send(getPureError(err));
|
||||||
|
});
|
||||||
|
//^=====--------------------------------------=====^//
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
app : app
|
||||||
|
};
|
||||||
37
server/db.js
Normal file
37
server/db.js
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
// The main purpose of this file is to provide an interface for database
|
||||||
|
// connection. Even though the code is quite simple and basically a tiny
|
||||||
|
// wrapper around mongoose package, it works as single point where
|
||||||
|
// database setup/config is performed and the interface provided here can be
|
||||||
|
// reused by both the main application and all tests which require database
|
||||||
|
// connection.
|
||||||
|
|
||||||
|
const Mongoose = require('mongoose');
|
||||||
|
|
||||||
|
const getMongoDBURL = (config)=>{
|
||||||
|
return config.get('mongodb_uri') ||
|
||||||
|
config.get('mongolab_uri') ||
|
||||||
|
'mongodb://localhost/homebrewery';
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleConnectionError = (error)=>{
|
||||||
|
if(error) {
|
||||||
|
console.error('Could not connect to a Mongo database: \n');
|
||||||
|
console.error(error);
|
||||||
|
console.error('\nIf you are running locally, make sure mongodb.exe is running and DB URL is configured properly');
|
||||||
|
process.exit(1); // non-zero exit code to indicate an error
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const disconnect = async ()=>{
|
||||||
|
return await Mongoose.disconnect();
|
||||||
|
};
|
||||||
|
|
||||||
|
const connect = async (config)=>{
|
||||||
|
return await Mongoose.connect(getMongoDBURL(config),
|
||||||
|
{ retryWrites: false }, handleConnectionError);
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
connect : connect,
|
||||||
|
disconnect : disconnect
|
||||||
|
};
|
||||||
@@ -96,7 +96,7 @@ GoogleActions = {
|
|||||||
const drive = google.drive({ version: 'v3', auth: oAuth2Client });
|
const drive = google.drive({ version: 'v3', auth: oAuth2Client });
|
||||||
|
|
||||||
const obj = await drive.files.list({
|
const obj = await drive.files.list({
|
||||||
pageSize : 100,
|
pageSize : 1000,
|
||||||
fields : 'nextPageToken, files(id, name, description, createdTime, modifiedTime, properties)',
|
fields : 'nextPageToken, files(id, name, description, createdTime, modifiedTime, properties)',
|
||||||
q : 'mimeType != \'application/vnd.google-apps.folder\' and trashed = false'
|
q : 'mimeType != \'application/vnd.google-apps.folder\' and trashed = false'
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ const cx = require('classnames');
|
|||||||
const DISMISS_KEY = 'dismiss_render_warning';
|
const DISMISS_KEY = 'dismiss_render_warning';
|
||||||
|
|
||||||
const RenderWarnings = createClass({
|
const RenderWarnings = createClass({
|
||||||
|
displayName : 'RenderWarnings',
|
||||||
getInitialState : function() {
|
getInitialState : function() {
|
||||||
return {
|
return {
|
||||||
warnings : {}
|
warnings : {}
|
||||||
|
|||||||
@@ -40,6 +40,7 @@ if(typeof navigator !== 'undefined'){
|
|||||||
}
|
}
|
||||||
|
|
||||||
const CodeEditor = createClass({
|
const CodeEditor = createClass({
|
||||||
|
displayName : 'CodeEditor',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
language : '',
|
language : '',
|
||||||
|
|||||||
@@ -8,7 +8,8 @@ const NaturalCritIcon = require('naturalcrit/svg/naturalcrit.svg.jsx');
|
|||||||
|
|
||||||
const Nav = {
|
const Nav = {
|
||||||
base : createClass({
|
base : createClass({
|
||||||
render : function(){
|
displayName : 'Nav.base',
|
||||||
|
render : function(){
|
||||||
return <nav>
|
return <nav>
|
||||||
<div className='navContent'>
|
<div className='navContent'>
|
||||||
{this.props.children}
|
{this.props.children}
|
||||||
@@ -26,7 +27,8 @@ const Nav = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
section : createClass({
|
section : createClass({
|
||||||
render : function(){
|
displayName : 'Nav.section',
|
||||||
|
render : function(){
|
||||||
return <div className='navSection'>
|
return <div className='navSection'>
|
||||||
{this.props.children}
|
{this.props.children}
|
||||||
</div>;
|
</div>;
|
||||||
@@ -34,6 +36,7 @@ const Nav = {
|
|||||||
}),
|
}),
|
||||||
|
|
||||||
item : createClass({
|
item : createClass({
|
||||||
|
displayName : 'Nav.item',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
icon : null,
|
icon : null,
|
||||||
@@ -69,6 +72,7 @@ const Nav = {
|
|||||||
}),
|
}),
|
||||||
|
|
||||||
dropdown : createClass({
|
dropdown : createClass({
|
||||||
|
displayName : 'Nav.dropdown',
|
||||||
getInitialState : function() {
|
getInitialState : function() {
|
||||||
return {
|
return {
|
||||||
showDropdown : false
|
showDropdown : false
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ const _ = require('lodash');
|
|||||||
const cx = require('classnames');
|
const cx = require('classnames');
|
||||||
|
|
||||||
const SplitPane = createClass({
|
const SplitPane = createClass({
|
||||||
|
displayName : 'SplitPane',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
storageKey : 'naturalcrit-pane-split',
|
storageKey : 'naturalcrit-pane-split',
|
||||||
@@ -77,6 +78,7 @@ const SplitPane = createClass({
|
|||||||
});
|
});
|
||||||
|
|
||||||
const Pane = createClass({
|
const Pane = createClass({
|
||||||
|
displayName : 'Pane',
|
||||||
getDefaultProps : function() {
|
getDefaultProps : function() {
|
||||||
return {
|
return {
|
||||||
width : null
|
width : null
|
||||||
|
|||||||
@@ -1,7 +0,0 @@
|
|||||||
const test = require('pico-check');
|
|
||||||
|
|
||||||
test('Just setting up a spot for future tests', (t)=>{
|
|
||||||
t.pass();
|
|
||||||
});
|
|
||||||
|
|
||||||
module.exports = test;
|
|
||||||
15
tests/markdown/basic.test.js
Normal file
15
tests/markdown/basic.test.js
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
/* eslint-disable max-lines */
|
||||||
|
|
||||||
|
const Markdown = require('naturalcrit/markdown.js');
|
||||||
|
|
||||||
|
test('Escapes <script> tag', function() {
|
||||||
|
const source = '<script></script>';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
expect(rendered).toMatch('<script></script>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Processes the markdown within an HTML block if its just a class wrapper', function() {
|
||||||
|
const source = '<div>*Bold text*</div>';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
expect(rendered).toBe('<div> <p><em>Bold text</em></p>\n </div>');
|
||||||
|
});
|
||||||
128
tests/markdown/mustache-span.test.js
Normal file
128
tests/markdown/mustache-span.test.js
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
/* eslint-disable max-lines */
|
||||||
|
|
||||||
|
const Markdown = require('naturalcrit/markdown.js');
|
||||||
|
|
||||||
|
test('Renders a mustache span with text only', function() {
|
||||||
|
const source = '{{ text}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
expect(rendered).toBe('<span class="inline-block ">text</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with text only, but with spaces', function() {
|
||||||
|
const source = '{{ this is a text}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
expect(rendered).toBe('<span class="inline-block ">this is a text</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders an empty mustache span', function() {
|
||||||
|
const source = '{{}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
expect(rendered).toBe('<span class="inline-block "></span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with just a space', function() {
|
||||||
|
const source = '{{ }}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
expect(rendered).toBe('<span class="inline-block "></span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with a few spaces only', function() {
|
||||||
|
const source = '{{ }}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
expect(rendered).toBe('<span class="inline-block "></span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with text and class', function() {
|
||||||
|
const source = '{{my-class text}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
// FIXME: why do we have those two extra spaces after closing "?
|
||||||
|
expect(rendered).toBe('<span class="inline-block my-class" >text</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with text and two classes', function() {
|
||||||
|
const source = '{{my-class,my-class2 text}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
// FIXME: why do we have those two extra spaces after closing "?
|
||||||
|
expect(rendered).toBe('<span class="inline-block my-class my-class2" >text</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with text with spaces and class', function() {
|
||||||
|
const source = '{{my-class this is a text}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
// FIXME: why do we have those two extra spaces after closing "?
|
||||||
|
expect(rendered).toBe('<span class="inline-block my-class" >this is a text</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with text and id', function() {
|
||||||
|
const source = '{{#my-span text}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
// FIXME: why do we have that one extra space after closing "?
|
||||||
|
expect(rendered).toBe('<span class="inline-block " id="my-span" >text</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with text and two ids', function() {
|
||||||
|
const source = '{{#my-span,#my-favorite-span text}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
// FIXME: do we need to report an error here somehow?
|
||||||
|
expect(rendered).toBe('<span class="inline-block " id="my-span" >text</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with text and css property', function() {
|
||||||
|
const source = '{{color:red text}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
expect(rendered).toBe('<span class="inline-block " style="color:red;">text</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with text and two css properties', function() {
|
||||||
|
const source = '{{color:red,padding:5px text}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
expect(rendered).toBe('<span class="inline-block " style="color:red; padding:5px;">text</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with text and css property which contains quotes', function() {
|
||||||
|
const source = '{{font:"trebuchet ms" text}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
// FIXME: is it correct to remove quotes surrounding css property value?
|
||||||
|
expect(rendered).toBe('<span class="inline-block " style="font:trebuchet ms;">text</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with text and two css properties which contains quotes', function() {
|
||||||
|
const source = '{{font:"trebuchet ms",padding:"5px 10px" text}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
expect(rendered).toBe('<span class="inline-block " style="font:trebuchet ms; padding:5px 10px;">text</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
test('Renders a mustache span with text with quotes and css property which contains quotes', function() {
|
||||||
|
const source = '{{font:"trebuchet ms" text "with quotes"}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
expect(rendered).toBe('<span class="inline-block " style="font:trebuchet ms;">text “with quotes”</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Renders a mustache span with text, id, class and a couple of css properties', function() {
|
||||||
|
const source = '{{pen,#author,color:orange,font-family:"trebuchet ms" text}}';
|
||||||
|
const rendered = Markdown.render(source);
|
||||||
|
expect(rendered).toBe('<span class="inline-block pen" id="author" style="color:orange; font-family:trebuchet ms;">text</span>');
|
||||||
|
});
|
||||||
|
|
||||||
|
// TODO: add tests for ID with accordance to CSS spec:
|
||||||
|
//
|
||||||
|
// From https://drafts.csswg.org/selectors/#id-selectors:
|
||||||
|
//
|
||||||
|
// > An ID selector consists of a “number sign” (U+0023, #) immediately followed by the ID value, which must be a CSS identifier.
|
||||||
|
//
|
||||||
|
// From: https://www.w3.org/TR/CSS21/syndata.html#value-def-identifier:
|
||||||
|
//
|
||||||
|
// > In CSS, identifiers (including element names, classes, and IDs in selectors) can contain only the characters [a-zA-Z0-9]
|
||||||
|
// > and ISO 10646 characters U+00A0 and higher, plus the hyphen (-) and the underscore (_);
|
||||||
|
// > they cannot start with a digit, two hyphens, or a hyphen followed by a digit.
|
||||||
|
// > Identifiers can also contain escaped characters and any ISO 10646 character as a numeric code (see next item).
|
||||||
|
// > For instance, the identifier "B&W?" may be written as "B\&W\?" or "B\26 W\3F".
|
||||||
|
// > Note that Unicode is code-by-code equivalent to ISO 10646 (see [UNICODE] and [ISO10646]).
|
||||||
|
|
||||||
|
// TODO: add tests for class with accordance to CSS spec:
|
||||||
|
//
|
||||||
|
// From: https://drafts.csswg.org/selectors/#class-html:
|
||||||
|
//
|
||||||
|
// > The class selector is given as a full stop (. U+002E) immediately followed by an identifier.
|
||||||
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
//Set up configs and DB connectiosna nd what not in here
|
|
||||||
Reference in New Issue
Block a user