mirror of
https://github.com/naturalcrit/homebrewery.git
synced 2025-12-24 18:32:41 +00:00
251 lines
6.7 KiB
JavaScript
251 lines
6.7 KiB
JavaScript
/* eslint-disable max-lines */
|
|
const _ = require('lodash');
|
|
const Markdown = require('marked');
|
|
const renderer = new Markdown.Renderer();
|
|
|
|
//Processes the markdown within an HTML block if it's just a class-wrapper
|
|
renderer.html = function (html) {
|
|
if(_.startsWith(_.trim(html), '<div') && _.endsWith(_.trim(html), '</div>')){
|
|
const openTag = html.substring(0, html.indexOf('>')+1);
|
|
html = html.substring(html.indexOf('>')+1);
|
|
html = html.substring(0, html.lastIndexOf('</div>'));
|
|
return `${openTag} ${Markdown(html)} </div>`;
|
|
}
|
|
return html;
|
|
};
|
|
|
|
// Don't wrap {{ Divs or {{ empty Spans in <p> tags
|
|
renderer.paragraph = function(text){
|
|
let match;
|
|
if(text.startsWith('<div') || text.startsWith('</div'))
|
|
return `${text}`;
|
|
else if(match = text.match(/(^|^.*?\n)<span class="inline([^>]*><\/span>)$/))
|
|
return `<p>${match[1]}</p><span class="inline-block"${match[2]}`;
|
|
else
|
|
return `<p>${text}</p>\n`;
|
|
};
|
|
|
|
// Mustache-style Divs {{class \n content ... \n}}
|
|
let blockCount = 0;
|
|
const blockRegex = /^ *{{(?:="[\w,\-. ]*"|[^"'\s])*$|^ *}}$/gm;
|
|
const inlineFullRegex = /{{[^\n]*}}/g;
|
|
const inlineRegex = /{{(?:="[\w,\-. ]*"|[^"'{}}\s])*\s*|}}/g;
|
|
|
|
renderer.text = function(text){
|
|
const newText = text.replaceAll('"', '"');
|
|
let matches;
|
|
|
|
if(matches = newText.match(inlineFullRegex)) {
|
|
|
|
//SPAN - INLINE
|
|
matches = newText.match(inlineRegex);
|
|
let matchIndex = 0;
|
|
const res = _.reduce(newText.split(inlineRegex), (r, splitText)=>{
|
|
|
|
if(splitText) r.push(Markdown.parseInline(splitText, { renderer: renderer }));
|
|
|
|
const block = matches[matchIndex] ? matches[matchIndex].trimLeft() : '';
|
|
if(block && block.startsWith('{{')) {
|
|
const values = processStyleTags(block.substring(2));
|
|
r.push(`<span class="inline ${values}>`);
|
|
blockCount++;
|
|
} else if(block == '}}' && blockCount !== 0){
|
|
r.push('</span>');
|
|
blockCount--;
|
|
}
|
|
|
|
matchIndex++;
|
|
|
|
return r;
|
|
}, []).join('');
|
|
return `${res}`;
|
|
} else if(matches = newText.match(blockRegex)) {
|
|
//DIV - BLOCK-LEVEL
|
|
let matchIndex = 0;
|
|
const res = _.reduce(newText.split(blockRegex), (r, splitText)=>{
|
|
if(splitText) r.push(Markdown.parseInline(splitText, { renderer: renderer }));
|
|
|
|
const block = matches[matchIndex] ? matches[matchIndex].trimLeft() : '';
|
|
if(block && block.startsWith('{')) {
|
|
const values = processStyleTags(block.substring(2));
|
|
r.push(`<div class="block ${values}">`);
|
|
blockCount++;
|
|
} else if(block == '}}' && blockCount !== 0){
|
|
r.push('</div>');
|
|
blockCount--;
|
|
}
|
|
|
|
matchIndex++;
|
|
|
|
return r;
|
|
}, []).join('');
|
|
return res;
|
|
} else {
|
|
if(!matches) {
|
|
return `${text}`;
|
|
}
|
|
}
|
|
};
|
|
|
|
//Fix local links in the Preview iFrame to link inside the frame
|
|
renderer.link = function (href, title, text) {
|
|
let self = false;
|
|
if(href[0] == '#') {
|
|
self = true;
|
|
}
|
|
href = cleanUrl(this.options.sanitize, this.options.baseUrl, href);
|
|
|
|
if(href === null) {
|
|
return text;
|
|
}
|
|
let out = `<a href="${escape(href)}"`;
|
|
if(title) {
|
|
out += ` title="${title}"`;
|
|
}
|
|
if(self) {
|
|
out += ' target="_self"';
|
|
}
|
|
out += `>${text}</a>`;
|
|
return out;
|
|
};
|
|
|
|
const nonWordAndColonTest = /[^\w:]/g;
|
|
const cleanUrl = function (sanitize, base, href) {
|
|
if(sanitize) {
|
|
let prot;
|
|
try {
|
|
prot = decodeURIComponent(unescape(href))
|
|
.replace(nonWordAndColonTest, '')
|
|
.toLowerCase();
|
|
} catch (e) {
|
|
return null;
|
|
}
|
|
if(prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0 || prot.indexOf('data:') === 0) {
|
|
return null;
|
|
}
|
|
}
|
|
try {
|
|
href = encodeURI(href).replace(/%25/g, '%');
|
|
} catch (e) {
|
|
return null;
|
|
}
|
|
return href;
|
|
};
|
|
|
|
const escapeTest = /[&<>"']/;
|
|
const escapeReplace = /[&<>"']/g;
|
|
const escapeTestNoEncode = /[<>"']|&(?!#?\w+;)/;
|
|
const escapeReplaceNoEncode = /[<>"']|&(?!#?\w+;)/g;
|
|
const escapeReplacements = {
|
|
'&' : '&',
|
|
'<' : '<',
|
|
'>' : '>',
|
|
'"' : '"',
|
|
'\'' : '''
|
|
};
|
|
const getEscapeReplacement = (ch)=>escapeReplacements[ch];
|
|
const escape = function (html, encode) {
|
|
if(encode) {
|
|
if(escapeTest.test(html)) {
|
|
return html.replace(escapeReplace, getEscapeReplacement);
|
|
}
|
|
} else {
|
|
if(escapeTestNoEncode.test(html)) {
|
|
return html.replace(escapeReplaceNoEncode, getEscapeReplacement);
|
|
}
|
|
}
|
|
return html;
|
|
};
|
|
|
|
const sanatizeScriptTags = (content)=>{
|
|
return content
|
|
.replace(/<script/ig, '<script')
|
|
.replace(/<\/script>/ig, '</script>');
|
|
};
|
|
|
|
const tagTypes = ['div', 'span', 'a'];
|
|
const tagRegex = new RegExp(`(${
|
|
_.map(tagTypes, (type)=>{
|
|
return `\\<${type}|\\</${type}>`;
|
|
}).join('|')})`, 'g');
|
|
|
|
const processStyleTags = (string)=>{
|
|
const tags = string.match(/(?:[^, "=]+|="[^"]*")+/g);
|
|
|
|
if(!tags) return '"';
|
|
|
|
const id = _.remove(tags, (tag)=>tag.startsWith('#')).map((tag)=>tag.slice(1))[0];
|
|
const classes = _.remove(tags, (tag)=>!tag.includes('"'));
|
|
const styles = tags.map((tag)=>tag.replace(/="(.*)"/g, ':$1;'));
|
|
return `${classes.join(' ')}" ${id ? `id="${id}"` : ''} ${styles ? `style="${styles.join(' ')}"` : ''}`;
|
|
};
|
|
|
|
module.exports = {
|
|
marked : Markdown,
|
|
render : (rawBrewText)=>{
|
|
blockCount = 0;
|
|
rawBrewText = rawBrewText.replace(/^\\column$/gm, `<div class='columnSplit'></div>`)
|
|
.replace(/^(:+)$/gm, (match)=>`${`<div class='blank'></div>`.repeat(match.length)}\n`)
|
|
.replace(/(?:^|>) *:([^:\n]*):([^\n]*)\n/gm, (match, term, def)=>`<dt>${Markdown.parseInline(term)}</dt><dd>${def}</dd>`)
|
|
.replace(/(<dt>.*<\/dt><dd>.*<\/dd>\n?)+/gm, `<dl>$1</dl>\n\n`)
|
|
.replace(/^}}/gm, '\n}}')
|
|
.replace(/^({{[^\n]*)$/gm, '$1\n');
|
|
return Markdown(
|
|
sanatizeScriptTags(rawBrewText),
|
|
{ renderer: renderer }
|
|
);
|
|
},
|
|
|
|
validate : (rawBrewText)=>{
|
|
const errors = [];
|
|
const leftovers = _.reduce(rawBrewText.split('\n'), (acc, line, _lineNumber)=>{
|
|
const lineNumber = _lineNumber + 1;
|
|
const matches = line.match(tagRegex);
|
|
if(!matches || !matches.length) return acc;
|
|
|
|
_.each(matches, (match)=>{
|
|
_.each(tagTypes, (type)=>{
|
|
if(match == `<${type}`){
|
|
acc.push({
|
|
type : type,
|
|
line : lineNumber
|
|
});
|
|
}
|
|
if(match === `</${type}>`){
|
|
if(!acc.length){
|
|
errors.push({
|
|
line : lineNumber,
|
|
type : type,
|
|
text : 'Unmatched closing tag',
|
|
id : 'CLOSE'
|
|
});
|
|
} else if(_.last(acc).type == type){
|
|
acc.pop();
|
|
} else {
|
|
errors.push({
|
|
line : `${_.last(acc).line} to ${lineNumber}`,
|
|
type : type,
|
|
text : 'Type mismatch on closing tag',
|
|
id : 'MISMATCH'
|
|
});
|
|
acc.pop();
|
|
}
|
|
}
|
|
});
|
|
});
|
|
return acc;
|
|
}, []);
|
|
|
|
_.each(leftovers, (unmatched)=>{
|
|
errors.push({
|
|
line : unmatched.line,
|
|
type : unmatched.type,
|
|
text : 'Unmatched opening tag',
|
|
id : 'OPEN'
|
|
});
|
|
});
|
|
|
|
return errors;
|
|
},
|
|
};
|