mirror of https://github.com/Requarks/wiki.git
30 changed files with 487 additions and 106 deletions
Unified View
Diff Options
-
62agent.js
-
2assets/css/app.css
-
BINassets/fonts/Roboto-Black.woff
-
BINassets/fonts/Roboto-BlackItalic.woff
-
BINassets/fonts/Roboto-Bold.woff
-
BINassets/fonts/Roboto-BoldItalic.woff
-
BINassets/fonts/Roboto-Light.woff
-
BINassets/fonts/Roboto-LightItalic.woff
-
BINassets/fonts/Roboto-Medium.woff
-
BINassets/fonts/Roboto-MediumItalic.woff
-
BINassets/fonts/Roboto-Regular.woff
-
BINassets/fonts/Roboto-RegularItalic.woff
-
BINassets/fonts/Roboto-Thin.woff
-
BINassets/fonts/Roboto-ThinItalic.woff
-
2assets/js/app.js
-
63assets/js/libs.js
-
26client/js/app.js
-
8client/scss/layout/_content.scss
-
11client/scss/layout/_header.scss
-
59config.sample.yml
-
1gulpfile.js
-
32lib/internalAuth.js
-
6models/entries.js
-
2models/git.js
-
82models/search.js
-
1package.json
-
24server.js
-
73views/common/header.pug
-
2views/layout.pug
-
137ws-server.js
2
assets/css/app.css
File diff suppressed because it is too large
View File
File diff suppressed because it is too large
View File
@ -1 +1 @@ |
|||||
"use strict";function _classCallCheck(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}function setInputSelection(e,t,a){if(e.focus(),"undefined"!=typeof e.selectionStart)e.selectionStart=t,e.selectionEnd=a;else if(document.selection&&document.selection.createRange){e.select();var n=document.selection.createRange();n.collapse(!0),n.moveEnd("character",a),n.moveStart("character",t),n.select()}}function makeSafePath(e){var t=_.split(_.trim(e),"/");return t=_.map(t,function(e){return _.kebabCase(_.deburr(_.trim(e)))}),_.join(_.filter(t,function(e){return!_.isEmpty(e)}),"/")}var _createClass=function(){function e(e,t){for(var a=0;a<t.length;a++){var n=t[a];n.enumerable=n.enumerable||!1,n.configurable=!0,"value"in n&&(n.writable=!0),Object.defineProperty(e,n.key,n)}}return function(t,a,n){return a&&e(t.prototype,a),n&&e(t,n),t}}(),Alerts=function(){function e(){_classCallCheck(this,e);var t=this;t.mdl=new Vue({el:"#alerts",data:{children:[]},methods:{acknowledge:function(e){t.close(e)}}}),t.uidNext=1}return _createClass(e,[{key:"push",value:function(e){var t=this,a=_.defaults(e,{_uid:t.uidNext,class:"is-info",message:"---",sticky:!1,title:"---"});t.mdl.children.push(a),a.sticky||_.delay(function(){t.close(a._uid)},5e3),t.uidNext++}},{key:"pushError",value:function(e,t){this.push({class:"is-danger",message:t,sticky:!1,title:e})}},{key:"pushSuccess",value:function(e,t){this.push({class:"is-success",message:t,sticky:!1,title:e})}},{key:"close",value:function(e){var t=this,a=_.findIndex(t.mdl.children,["_uid",e]),n=_.nth(t.mdl.children,a);a>=0&&n&&(n.class+=" exit",t.mdl.children.$set(a,n),_.delay(function(){t.mdl.children.$remove(n)},500))}}]),e}();jQuery(document).ready(function(e){e("a").smoothScroll({speed:400,offset:-20});new Sticky(".stickyscroll");e(window).bind("beforeunload",function(){e("#notifload").addClass("active")}),e(document).ajaxSend(function(){e("#notifload").addClass("active")}).ajaxComplete(function(){e("#notifload").removeClass("active")});var t=new Alerts;if(alertsData&&_.forEach(alertsData,function(e){t.push(e)}),1===e("#mk-editor").length)var a=new SimpleMDE({autofocus:!0,autoDownloadFontAwesome:!1,element:e("#mk-editor").get(0),hideIcons:["heading","quote"],placeholder:"Enter Markdown formatted content here...",showIcons:["strikethrough","heading-1","heading-2","heading-3","code","table","horizontal-rule"],spellChecker:!1,status:!1});if(e("#page-type-view").length&&!function(){var t="home"!==e("#page-type-view").data("entrypath")?e("#page-type-view").data("entrypath")+"/":"",a=t+"new-page";e(".btn-create-prompt").on("click",function(n){e("#txt-create-prompt").val(a),e("#modal-create-prompt").toggleClass("is-active"),setInputSelection(e("#txt-create-prompt").get(0),t.length,a.length),e("#txt-create-prompt").removeClass("is-danger").next().addClass("is-hidden")}),e("#txt-create-prompt").on("keypress",function(t){13===t.which&&e(".btn-create-go").trigger("click")}),e(".btn-create-go").on("click",function(t){var a=makeSafePath(e("#txt-create-prompt").val());_.isEmpty(a)?e("#txt-create-prompt").addClass("is-danger").next().removeClass("is-hidden"):(e("#txt-create-prompt").parent().addClass("is-loading"),window.location.assign("/create/"+a))})}(),e("#page-type-create").length&&(e(".btn-create-discard").on("click",function(t){e("#modal-create-discard").toggleClass("is-active")}),e(".btn-create-save").on("click",function(n){e.ajax(window.location.href,{data:{markdown:a.value()},dataType:"json",method:"PUT"}).then(function(a,n,o){a.ok?window.location.assign("/"+e("#page-type-create").data("entrypath")):t.pushError("Something went wrong",a.error)},function(e,a,n){t.pushError("Something went wrong","Save operation failed.")})})),e("#page-type-edit").length){e(".editor-toolbar").attr("data-margin-top",e("#header").height());new Sticky(".editor-toolbar");e(".btn-edit-discard").on("click",function(t){e("#modal-edit-discard").toggleClass("is-active")}),e(".btn-edit-save").on("click",function(n){e.ajax(window.location.href,{data:{markdown:a.value()},dataType:"json",method:"PUT"}).then(function(a,n,o){a.ok?window.location.assign("/"+e("#page-type-edit").data("entrypath")):t.pushError("Something went wrong",a.error)},function(e,a,n){t.pushError("Something went wrong","Save operation failed.")})})}if(e("#page-type-source").length){var n=ace.edit("source-display");n.setTheme("ace/theme/tomorrow_night"),n.getSession().setMode("ace/mode/markdown"),n.setReadOnly(!0),n.renderer.updateFull()}}); |
"use strict";function _classCallCheck(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}function setInputSelection(e,t,a){if(e.focus(),"undefined"!=typeof e.selectionStart)e.selectionStart=t,e.selectionEnd=a;else if(document.selection&&document.selection.createRange){e.select();var n=document.selection.createRange();n.collapse(!0),n.moveEnd("character",a),n.moveStart("character",t),n.select()}}function makeSafePath(e){var t=_.split(_.trim(e),"/");return t=_.map(t,function(e){return _.kebabCase(_.deburr(_.trim(e)))}),_.join(_.filter(t,function(e){return!_.isEmpty(e)}),"/")}var _createClass=function(){function e(e,t){for(var a=0;a<t.length;a++){var n=t[a];n.enumerable=n.enumerable||!1,n.configurable=!0,"value"in n&&(n.writable=!0),Object.defineProperty(e,n.key,n)}}return function(t,a,n){return a&&e(t.prototype,a),n&&e(t,n),t}}(),Alerts=function(){function e(){_classCallCheck(this,e);var t=this;t.mdl=new Vue({el:"#alerts",data:{children:[]},methods:{acknowledge:function(e){t.close(e)}}}),t.uidNext=1}return _createClass(e,[{key:"push",value:function(e){var t=this,a=_.defaults(e,{_uid:t.uidNext,class:"is-info",message:"---",sticky:!1,title:"---"});t.mdl.children.push(a),a.sticky||_.delay(function(){t.close(a._uid)},5e3),t.uidNext++}},{key:"pushError",value:function(e,t){this.push({class:"is-danger",message:t,sticky:!1,title:e})}},{key:"pushSuccess",value:function(e,t){this.push({class:"is-success",message:t,sticky:!1,title:e})}},{key:"close",value:function(e){var t=this,a=_.findIndex(t.mdl.children,["_uid",e]),n=_.nth(t.mdl.children,a);a>=0&&n&&(n.class+=" exit",t.mdl.children.$set(a,n),_.delay(function(){t.mdl.children.$remove(n)},500))}}]),e}();jQuery(document).ready(function(e){e("a").smoothScroll({speed:400,offset:-20});new Sticky(".stickyscroll");e(window).bind("beforeunload",function(){e("#notifload").addClass("active")}),e(document).ajaxSend(function(){e("#notifload").addClass("active")}).ajaxComplete(function(){e("#notifload").removeClass("active")});var t=new Alerts;if(alertsData&&_.forEach(alertsData,function(e){t.push(e)}),1===e("#mk-editor").length)var a=new SimpleMDE({autofocus:!0,autoDownloadFontAwesome:!1,element:e("#mk-editor").get(0),hideIcons:["heading","quote"],placeholder:"Enter Markdown formatted content here...",showIcons:["strikethrough","heading-1","heading-2","heading-3","code","table","horizontal-rule"],spellChecker:!1,status:!1});var n=io(ioHost),o=new Vue({el:"#header-container",data:{searchq:"",searchres:[]},watch:{searchq:function(e,t){e.length>=3&&n.emit("search",{terms:e},function(e){o.$set("searchres",e)})}},methods:{}});if(e("#page-type-view").length&&!function(){var t="home"!==e("#page-type-view").data("entrypath")?e("#page-type-view").data("entrypath")+"/":"",a=t+"new-page";e(".btn-create-prompt").on("click",function(n){e("#txt-create-prompt").val(a),e("#modal-create-prompt").toggleClass("is-active"),setInputSelection(e("#txt-create-prompt").get(0),t.length,a.length),e("#txt-create-prompt").removeClass("is-danger").next().addClass("is-hidden")}),e("#txt-create-prompt").on("keypress",function(t){13===t.which&&e(".btn-create-go").trigger("click")}),e(".btn-create-go").on("click",function(t){var a=makeSafePath(e("#txt-create-prompt").val());_.isEmpty(a)?e("#txt-create-prompt").addClass("is-danger").next().removeClass("is-hidden"):(e("#txt-create-prompt").parent().addClass("is-loading"),window.location.assign("/create/"+a))})}(),e("#page-type-create").length&&(e(".btn-create-discard").on("click",function(t){e("#modal-create-discard").toggleClass("is-active")}),e(".btn-create-save").on("click",function(n){e.ajax(window.location.href,{data:{markdown:a.value()},dataType:"json",method:"PUT"}).then(function(a,n,o){a.ok?window.location.assign("/"+e("#page-type-create").data("entrypath")):t.pushError("Something went wrong",a.error)},function(e,a,n){t.pushError("Something went wrong","Save operation failed.")})})),e("#page-type-edit").length){e(".editor-toolbar").attr("data-margin-top",e("#header").height());new Sticky(".editor-toolbar");e(".btn-edit-discard").on("click",function(t){e("#modal-edit-discard").toggleClass("is-active")}),e(".btn-edit-save").on("click",function(n){e.ajax(window.location.href,{data:{markdown:a.value()},dataType:"json",method:"PUT"}).then(function(a,n,o){a.ok?window.location.assign("/"+e("#page-type-edit").data("entrypath")):t.pushError("Something went wrong",a.error)},function(e,a,n){t.pushError("Something went wrong","Save operation failed.")})})}if(e("#page-type-source").length){var i=ace.edit("source-display");i.setTheme("ace/theme/tomorrow_night"),i.getSession().setMode("ace/mode/markdown"),i.setReadOnly(!0),i.renderer.updateFull()}}); |
63
assets/js/libs.js
File diff suppressed because it is too large
View File
File diff suppressed because it is too large
View File
@ -0,0 +1,32 @@ |
|||||
|
"use strict"; |
||||
|
|
||||
|
const crypto = require('crypto'); |
||||
|
|
||||
|
/** |
||||
|
* Internal Authentication |
||||
|
*/ |
||||
|
module.exports = { |
||||
|
|
||||
|
_curKey: false, |
||||
|
|
||||
|
init(inKey) { |
||||
|
|
||||
|
this._curKey = inKey; |
||||
|
|
||||
|
return this; |
||||
|
|
||||
|
}, |
||||
|
|
||||
|
generateKey() { |
||||
|
|
||||
|
return crypto.randomBytes(20).toString('hex') |
||||
|
|
||||
|
}, |
||||
|
|
||||
|
validateKey(inKey) { |
||||
|
|
||||
|
return inKey === this._curKey; |
||||
|
|
||||
|
} |
||||
|
|
||||
|
}; |
@ -1,34 +1,45 @@ |
|||||
|
|
||||
nav.nav.has-shadow.stickyscroll#header |
#header-container |
||||
.nav-left |
nav.nav.has-shadow.stickyscroll#header |
||||
block rootNavLeft |
.nav-left |
||||
a.nav-item.is-brand(href='/') |
block rootNavLeft |
||||
img(src='/favicons/android-icon-96x96.png', alt='Wiki') |
a.nav-item.is-brand(href='/') |
||||
a.nav-item(href='/') |
img(src='/favicons/android-icon-96x96.png', alt='Wiki') |
||||
h1.title Wiki |
a.nav-item(href='/') |
||||
.nav-center |
h1.title Wiki |
||||
block rootNavCenter |
.nav-center |
||||
p.nav-item |
block rootNavCenter |
||||
input.input(type='text', placeholder='Search...', style= { 'max-width': '300px', width: '33vw' }) |
p.nav-item |
||||
span.nav-toggle |
input.input(type='text', v-model='searchq', debounce='500' placeholder='Search...', style= { 'max-width': '300px', width: '33vw' }) |
||||
span |
span.nav-toggle |
||||
span |
span |
||||
span |
span |
||||
.nav-right.nav-menu |
span |
||||
block rootNavRight |
.nav-right.nav-menu |
||||
i.nav-item#notifload |
block rootNavRight |
||||
a.nav-item(href='/history/' + pageData.meta.path) |
i.nav-item#notifload |
||||
| History |
a.nav-item(href='/history/' + pageData.meta.path) |
||||
a.nav-item(href='/source/' + pageData.meta.path) |
| History |
||||
| Source |
a.nav-item(href='/source/' + pageData.meta.path) |
||||
span.nav-item |
| Source |
||||
a.button(href='/edit/' + pageData.meta.path) |
span.nav-item |
||||
span.icon |
a.button(href='/edit/' + pageData.meta.path) |
||||
i.fa.fa-edit |
span.icon |
||||
span Edit |
i.fa.fa-edit |
||||
a.button.is-primary.btn-create-prompt |
span Edit |
||||
span.icon |
a.button.is-primary.btn-create-prompt |
||||
i.fa.fa-plus |
span.icon |
||||
span Create |
i.fa.fa-plus |
||||
|
span Create |
||||
|
|
||||
|
.box.searchresults |
||||
|
.menu |
||||
|
p.menu-label |
||||
|
| Search Results |
||||
|
ul.menu-list |
||||
|
li(v-for='sres in searchres') |
||||
|
a(href='#') {{ sres.document.title }} |
||||
|
p.menu-label |
||||
|
| Do you mean...? |
||||
|
|
||||
|
|
@ -0,0 +1,137 @@ |
|||||
|
// ===========================================
|
||||
|
// REQUARKS WIKI - WebSocket Server
|
||||
|
// 1.0.0
|
||||
|
// Licensed under AGPLv3
|
||||
|
// ===========================================
|
||||
|
|
||||
|
global.ROOTPATH = __dirname; |
||||
|
|
||||
|
// ----------------------------------------
|
||||
|
// Load Winston
|
||||
|
// ----------------------------------------
|
||||
|
|
||||
|
var _isDebug = process.env.NODE_ENV === 'development'; |
||||
|
|
||||
|
global.winston = require('winston'); |
||||
|
winston.remove(winston.transports.Console) |
||||
|
winston.add(winston.transports.Console, { |
||||
|
level: (_isDebug) ? 'info' : 'warn', |
||||
|
prettyPrint: true, |
||||
|
colorize: true, |
||||
|
silent: false, |
||||
|
timestamp: true |
||||
|
}); |
||||
|
|
||||
|
// ----------------------------------------
|
||||
|
// Fetch internal handshake key
|
||||
|
// ----------------------------------------
|
||||
|
|
||||
|
if(!process.argv[2] || process.argv[2].length !== 40) { |
||||
|
winston.error('[WS] Illegal process start. Missing handshake key.'); |
||||
|
process.exit(1); |
||||
|
} |
||||
|
global.internalAuth = require('./lib/internalAuth').init(process.argv[2]);; |
||||
|
|
||||
|
// ----------------------------------------
|
||||
|
// Load modules
|
||||
|
// ----------------------------------------
|
||||
|
|
||||
|
winston.info('[WS] WS Server is initializing...'); |
||||
|
|
||||
|
var appconfig = require('./models/config')('./config.yml'); |
||||
|
|
||||
|
global.entries = require('./models/entries').init(appconfig); |
||||
|
global.mark = require('./models/markdown'); |
||||
|
global.search = require('./models/search').init(appconfig); |
||||
|
|
||||
|
// ----------------------------------------
|
||||
|
// Load modules
|
||||
|
// ----------------------------------------
|
||||
|
|
||||
|
var _ = require('lodash'); |
||||
|
var express = require('express'); |
||||
|
var path = require('path'); |
||||
|
var http = require('http'); |
||||
|
var socketio = require('socket.io'); |
||||
|
var moment = require('moment'); |
||||
|
|
||||
|
// ----------------------------------------
|
||||
|
// Define Express App
|
||||
|
// ----------------------------------------
|
||||
|
|
||||
|
global.app = express(); |
||||
|
|
||||
|
// ----------------------------------------
|
||||
|
// Controllers
|
||||
|
// ----------------------------------------
|
||||
|
|
||||
|
app.get('/', function(req, res){ |
||||
|
res.send('Requarks Wiki WebSocket server'); |
||||
|
}); |
||||
|
|
||||
|
// ----------------------------------------
|
||||
|
// Start WebSocket server
|
||||
|
// ----------------------------------------
|
||||
|
|
||||
|
winston.info('[SERVER] Starting WebSocket server on port ' + appconfig.wsPort + '...'); |
||||
|
|
||||
|
app.set('port', appconfig.wsPort); |
||||
|
var server = http.Server(app); |
||||
|
var io = socketio(server); |
||||
|
|
||||
|
server.on('error', (error) => { |
||||
|
if (error.syscall !== 'listen') { |
||||
|
throw error; |
||||
|
} |
||||
|
|
||||
|
switch (error.code) { |
||||
|
case 'EACCES': |
||||
|
console.error('Listening on port ' + appconfig.port + ' requires elevated privileges!'); |
||||
|
process.exit(1); |
||||
|
break; |
||||
|
case 'EADDRINUSE': |
||||
|
console.error('Port ' + appconfig.port + ' is already in use!'); |
||||
|
process.exit(1); |
||||
|
break; |
||||
|
default: |
||||
|
throw error; |
||||
|
} |
||||
|
}); |
||||
|
|
||||
|
server.listen(appconfig.wsPort, () => { |
||||
|
winston.info('[WS] WebSocket server started successfully! [RUNNING]'); |
||||
|
}); |
||||
|
|
||||
|
io.on('connection', (socket) => { |
||||
|
|
||||
|
socket.on('searchAdd', (data) => { |
||||
|
if(internalAuth.validateKey(data.auth)) { |
||||
|
search.add(data.content); |
||||
|
} |
||||
|
}); |
||||
|
|
||||
|
socket.on('search', (data, cb) => { |
||||
|
search.find(data.terms).then((results) => { |
||||
|
cb(results); |
||||
|
}); |
||||
|
}) |
||||
|
|
||||
|
}); |
||||
|
|
||||
|
/*setTimeout(() => { |
||||
|
search._si.searchAsync({ query: { AND: [{'*': ['unit']}] }}).then((stuff) => { console.log(stuff.hits); }); |
||||
|
}, 8000);*/ |
||||
|
|
||||
|
// ----------------------------------------
|
||||
|
// Shutdown gracefully
|
||||
|
// ----------------------------------------
|
||||
|
|
||||
|
process.on('disconnect', () => { |
||||
|
winston.warn('[WS] Lost connection to main server. Exiting... [' + moment().toISOString() + ']'); |
||||
|
server.close(); |
||||
|
process.exit(); |
||||
|
}); |
||||
|
|
||||
|
process.on('exit', () => { |
||||
|
server.stop(); |
||||
|
}); |
xxxxxxxxxx