Compare commits
15 Commits
Author | SHA1 | Date | |
---|---|---|---|
b98fbddd9e | |||
0818f57131 | |||
4a43aaabb6 | |||
a297829ffc | |||
4da2877a89 | |||
2e309b29f1 | |||
0cea6eb4ca | |||
d1268fe708 | |||
ff92316e1e | |||
629a7df3c4 | |||
30e23caf7f | |||
9afe8c5391 | |||
e958080702 | |||
6455f3ff10 | |||
1536c0721d |
2
.gitignore
vendored
2
.gitignore
vendored
@ -8,3 +8,5 @@ public/js/
|
|||||||
# Local data storage
|
# Local data storage
|
||||||
data/
|
data/
|
||||||
|
|
||||||
|
# Local config
|
||||||
|
config/local.json
|
||||||
|
@ -3,6 +3,7 @@ pipeline:
|
|||||||
image: node:18
|
image: node:18
|
||||||
commands:
|
commands:
|
||||||
- npm install
|
- npm install
|
||||||
|
- npm run grunt
|
||||||
|
|
||||||
lint:
|
lint:
|
||||||
image: node:18
|
image: node:18
|
||||||
|
55
Gruntfile.js
55
Gruntfile.js
@ -4,6 +4,51 @@ module.exports = function(grunt) {
|
|||||||
grunt.initConfig({
|
grunt.initConfig({
|
||||||
pkg: grunt.file.readJSON('package.json'),
|
pkg: grunt.file.readJSON('package.json'),
|
||||||
|
|
||||||
|
sass: {
|
||||||
|
dist: {
|
||||||
|
options: {
|
||||||
|
style: 'compressed'
|
||||||
|
},
|
||||||
|
files: [{
|
||||||
|
expand: true,
|
||||||
|
cwd: 'assets/styles',
|
||||||
|
src: ['**/*.scss'],
|
||||||
|
dest: 'public/css',
|
||||||
|
ext: '.css'
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
uglify: {
|
||||||
|
options: {
|
||||||
|
mangle: false
|
||||||
|
},
|
||||||
|
compile: {
|
||||||
|
files: {
|
||||||
|
'public/js/nechryael.min.js': ['assets/js/**/*.js']
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
watch: {
|
||||||
|
css: {
|
||||||
|
files: ['assets/styles/**/*.scss'],
|
||||||
|
tasks: ['sass'],
|
||||||
|
options: {
|
||||||
|
atBegin: true,
|
||||||
|
spawn: false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
js: {
|
||||||
|
files: ['assets/js/**/*.js'],
|
||||||
|
tasks: ['uglify'],
|
||||||
|
options: {
|
||||||
|
atBegin: true,
|
||||||
|
spawn: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
deb_package: {
|
deb_package: {
|
||||||
options: {
|
options: {
|
||||||
maintainer: 'Gregory Ballantine <gballantine@bitgoblin.tech>',
|
maintainer: 'Gregory Ballantine <gballantine@bitgoblin.tech>',
|
||||||
@ -39,7 +84,7 @@ module.exports = function(grunt) {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
cwd: './build/etc/',
|
cwd: './build/etc/',
|
||||||
src: 'production.json',
|
src: 'default.json',
|
||||||
dest: '/etc/overseer/'
|
dest: '/etc/overseer/'
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -75,10 +120,14 @@ module.exports = function(grunt) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Load the plugin that provides the "uglify" task.
|
// Load plugins.
|
||||||
|
grunt.loadNpmTasks('grunt-contrib-watch');
|
||||||
|
grunt.loadNpmTasks('grunt-contrib-sass');
|
||||||
|
grunt.loadNpmTasks('grunt-contrib-uglify');
|
||||||
grunt.loadNpmTasks('grunt-deb');
|
grunt.loadNpmTasks('grunt-deb');
|
||||||
|
|
||||||
// Default task(s).
|
// CLI tasks.
|
||||||
|
grunt.registerTask('default', ['sass', 'uglify']);
|
||||||
grunt.registerTask('package', ['deb_package']);
|
grunt.registerTask('package', ['deb_package']);
|
||||||
|
|
||||||
};
|
};
|
||||||
|
3
assets/js/nechryael.js
Normal file
3
assets/js/nechryael.js
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
$(document).ready(function() {
|
||||||
|
console.log('Document is ready!');
|
||||||
|
});
|
@ -1,4 +1,8 @@
|
|||||||
{
|
{
|
||||||
|
"server": {
|
||||||
|
"address": "0.0.0.0",
|
||||||
|
"port": 3000
|
||||||
|
},
|
||||||
"database": {
|
"database": {
|
||||||
"driver": "sqlite",
|
"driver": "sqlite",
|
||||||
"connection_string": "/opt/overseer/data/overseer.db"
|
"connection_string": "/opt/overseer/data/overseer.db"
|
@ -4,6 +4,7 @@ Description=Overseer inventory tracking app
|
|||||||
[Service]
|
[Service]
|
||||||
User=overseer
|
User=overseer
|
||||||
Group=overseer
|
Group=overseer
|
||||||
|
Environment="NODE_CONFIG_DIR=/etc/overseer"
|
||||||
ExecStart=/usr/bin/overseer
|
ExecStart=/usr/bin/overseer
|
||||||
SuccessExitStatus=143
|
SuccessExitStatus=143
|
||||||
|
|
||||||
|
@ -1,6 +1,15 @@
|
|||||||
{
|
{
|
||||||
|
"server": {
|
||||||
|
"address": "0.0.0.0",
|
||||||
|
"port": 3000
|
||||||
|
},
|
||||||
"database": {
|
"database": {
|
||||||
"driver": "sqlite",
|
"driver": "sqlite",
|
||||||
"connection_string": "data/overseer.db"
|
"connection_string": "data/overseer.db"
|
||||||
|
},
|
||||||
|
"redis": {
|
||||||
|
"host": "192.168.1.10",
|
||||||
|
"port": 6379,
|
||||||
|
"number": "0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
17
gulpfile.js
17
gulpfile.js
@ -1,17 +0,0 @@
|
|||||||
const gulp = require('gulp');
|
|
||||||
const { watch } = require('gulp');
|
|
||||||
const sass = require('gulp-sass')(require('sass'));
|
|
||||||
|
|
||||||
// compile
|
|
||||||
function styles(cb) {
|
|
||||||
return gulp.src('./assets/styles/**/*.scss')
|
|
||||||
.pipe(sass({outputStyle: 'compressed'}).on('error', sass.logError))
|
|
||||||
.pipe(gulp.dest('./public/css'));
|
|
||||||
cb();
|
|
||||||
}
|
|
||||||
|
|
||||||
// by default, watch files
|
|
||||||
exports.default = function() {
|
|
||||||
// compile sass stylesheets
|
|
||||||
watch('assets/styles/**/*.scss', styles);
|
|
||||||
};
|
|
26
index.js
26
index.js
@ -1,8 +1,11 @@
|
|||||||
const express = require('express');
|
const express = require('express');
|
||||||
|
const session = require('express-session');
|
||||||
|
const RedisStore = require('connect-redis')(session);
|
||||||
|
// const flash = require('express-flasher');
|
||||||
|
|
||||||
// instantiate new express.js app
|
// instantiate new express.js app
|
||||||
const app = express();
|
const app = express();
|
||||||
const port = 3000;
|
const config = require('config');
|
||||||
|
|
||||||
// initialize database connection
|
// initialize database connection
|
||||||
(async () => {
|
(async () => {
|
||||||
@ -12,6 +15,23 @@ const port = 3000;
|
|||||||
});
|
});
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
// initialize Redis store for session data
|
||||||
|
const redisClient = require('./src/redis');
|
||||||
|
|
||||||
|
// initialize express.js session w/ Redis datastore
|
||||||
|
app.use(session({
|
||||||
|
store: new RedisStore({
|
||||||
|
client: redisClient,
|
||||||
|
}), // use Redis datastore
|
||||||
|
resave: false, // don't save session if unmodified
|
||||||
|
saveUninitialized: false, // don't create session until something stored
|
||||||
|
secret: 'lord of the rings',
|
||||||
|
}));
|
||||||
|
|
||||||
|
// setup flash messaging
|
||||||
|
// app.use(flash.flash());
|
||||||
|
// app.use(flash.flashRead());
|
||||||
|
|
||||||
// set up body POST parameters
|
// set up body POST parameters
|
||||||
app.use(express.json());
|
app.use(express.json());
|
||||||
app.use(express.urlencoded({
|
app.use(express.urlencoded({
|
||||||
@ -37,6 +57,6 @@ app.get('/item/:id/edit', itemRoutes.getItemEdit);
|
|||||||
app.post('/item/:id/edit', itemRoutes.postItemEdit);
|
app.post('/item/:id/edit', itemRoutes.postItemEdit);
|
||||||
|
|
||||||
// start app
|
// start app
|
||||||
app.listen(port, () => {
|
app.listen(config.get('server.port'), config.get('server.address'), () => {
|
||||||
console.log(`Example app listening on port ${port}`);
|
console.log(`Overseer is listening on port ${config.get('server.port')}.`);
|
||||||
});
|
});
|
||||||
|
7444
package-lock.json
generated
7444
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
20
package.json
20
package.json
@ -1,12 +1,11 @@
|
|||||||
{
|
{
|
||||||
"name": "overseer",
|
"name": "overseer",
|
||||||
"version": "0.1.1",
|
"version": "0.2.0",
|
||||||
"description": "Self-hosted inventory tracker",
|
"description": "Self-hosted inventory tracker",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "node index.js",
|
"start": "node index.js",
|
||||||
"grunt": "grunt",
|
"grunt": "grunt",
|
||||||
"gulp": "gulp",
|
|
||||||
"nodemon": "nodemon index.js",
|
"nodemon": "nodemon index.js",
|
||||||
"lint": "eslint index.js src/**/*.js",
|
"lint": "eslint index.js src/**/*.js",
|
||||||
"test": "echo \"Error: no test specified\" && exit 1"
|
"test": "echo \"Error: no test specified\" && exit 1"
|
||||||
@ -21,7 +20,10 @@
|
|||||||
],
|
],
|
||||||
"author": "Gregory Ballanine <gballantine@bitgoblin.tech>",
|
"author": "Gregory Ballanine <gballantine@bitgoblin.tech>",
|
||||||
"uploaders": [
|
"uploaders": [
|
||||||
{ "name": "Gregory Ballantine", "email": "gballantine@bitgoblin.tech" }
|
{
|
||||||
|
"name": "Gregory Ballantine",
|
||||||
|
"email": "gballantine@bitgoblin.tech"
|
||||||
|
}
|
||||||
],
|
],
|
||||||
"license": "BSD-2-Clause",
|
"license": "BSD-2-Clause",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
@ -29,15 +31,23 @@
|
|||||||
"eslint-config-google": "^0.14.0",
|
"eslint-config-google": "^0.14.0",
|
||||||
"grunt": "^1.5.3",
|
"grunt": "^1.5.3",
|
||||||
"grunt-cli": "^1.4.3",
|
"grunt-cli": "^1.4.3",
|
||||||
|
"grunt-contrib-sass": "^2.0.0",
|
||||||
|
"grunt-contrib-uglify": "^5.2.2",
|
||||||
|
"grunt-contrib-watch": "^1.1.0",
|
||||||
"grunt-deb": "^0.2.5",
|
"grunt-deb": "^0.2.5",
|
||||||
"gulp": "^4.0.2",
|
|
||||||
"gulp-sass": "^5.1.0",
|
|
||||||
"nodemon": "^2.0.20",
|
"nodemon": "^2.0.20",
|
||||||
"sass": "^1.55.0"
|
"sass": "^1.55.0"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"config": "^3.3.8",
|
"config": "^3.3.8",
|
||||||
|
"connect-redis": "^6.1.3",
|
||||||
"express": "^4.18.2",
|
"express": "^4.18.2",
|
||||||
|
"express-session": "^1.17.3",
|
||||||
|
"mariadb": "^3.0.2",
|
||||||
|
"mysql2": "^2.3.3",
|
||||||
|
"pg": "^8.8.0",
|
||||||
|
"pg-hstore": "^2.3.4",
|
||||||
|
"redis": "^4.4.0",
|
||||||
"sequelize": "^6.25.3",
|
"sequelize": "^6.25.3",
|
||||||
"sqlite3": "^5.1.2",
|
"sqlite3": "^5.1.2",
|
||||||
"twig": "^1.15.4"
|
"twig": "^1.15.4"
|
||||||
|
@ -1,10 +1,7 @@
|
|||||||
const dbConfig = require('config').get('database');
|
const dbConfig = require('config').get('database');
|
||||||
const Sequelize = require('sequelize');
|
const Sequelize = require('sequelize');
|
||||||
|
|
||||||
const sequelize = new Sequelize({
|
const sequelize = initDatabase();
|
||||||
dialect: dbConfig.get('driver'),
|
|
||||||
storage: dbConfig.get('connection_string'),
|
|
||||||
});
|
|
||||||
|
|
||||||
const db = {};
|
const db = {};
|
||||||
|
|
||||||
@ -14,3 +11,29 @@ db.sequelize = sequelize;
|
|||||||
db.items = require('./item.js')(sequelize, Sequelize);
|
db.items = require('./item.js')(sequelize, Sequelize);
|
||||||
|
|
||||||
module.exports = db;
|
module.exports = db;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initializes a sequelize database connection.
|
||||||
|
*
|
||||||
|
* @return {object} - sequelize connection
|
||||||
|
*/
|
||||||
|
function initDatabase() {
|
||||||
|
let sequelize = null;
|
||||||
|
|
||||||
|
if (dbConfig.get('driver') == 'sqlite') {
|
||||||
|
sequelize = new Sequelize({
|
||||||
|
dialect: dbConfig.get('driver'),
|
||||||
|
storage: dbConfig.get('connection_string'),
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
const dbName = dbConfig.get('name');
|
||||||
|
const dbUsername = dbConfig.get('username');
|
||||||
|
const dbPassword = dbConfig.get('password');
|
||||||
|
sequelize = new Sequelize(dbName, dbUsername, dbPassword, {
|
||||||
|
dialect: dbConfig.get('driver'),
|
||||||
|
host: dbConfig.get('address'),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return sequelize;
|
||||||
|
}
|
||||||
|
31
src/redis.js
Normal file
31
src/redis.js
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
const redisConfig = require('config').get('redis');
|
||||||
|
|
||||||
|
exports.default = function() {
|
||||||
|
let redisUrl = 'redis://';
|
||||||
|
|
||||||
|
// add the redis username if defined
|
||||||
|
if (typeof redisConfig.get('username') !== 'undefined') {
|
||||||
|
redisUrl += redisConfig.get('username');
|
||||||
|
}
|
||||||
|
// add the user password if defined
|
||||||
|
if (typeof redisConfig.get('password') !== 'undefined') {
|
||||||
|
redisUrl += ':' + redisConfig.get('password') + '@';
|
||||||
|
}
|
||||||
|
// add redis host URL
|
||||||
|
redisUrl += redisConfig.get('host');
|
||||||
|
// add redis host port
|
||||||
|
redisUrl += ':' + redisConfig.get('port');
|
||||||
|
// add redis database number if defined
|
||||||
|
if (typeof redisConfig.get('number') !== 'undefined') {
|
||||||
|
redisUrl += redisConfig.get('number');
|
||||||
|
}
|
||||||
|
|
||||||
|
const { createClient } = require("redis");
|
||||||
|
let redisClient = createClient({
|
||||||
|
url: redisUrl,
|
||||||
|
legacyMode: true,
|
||||||
|
});
|
||||||
|
redisClient.connect().catch(console.error);
|
||||||
|
|
||||||
|
return redisClient;
|
||||||
|
};
|
@ -10,6 +10,8 @@ exports.getIndex = async function(req, res) {
|
|||||||
],
|
],
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// req.flash('info', 'This is a test flash message.');
|
||||||
|
|
||||||
res.render('index.twig', {
|
res.render('index.twig', {
|
||||||
inventory: items,
|
inventory: items,
|
||||||
});
|
});
|
||||||
|
@ -9,7 +9,7 @@
|
|||||||
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/skeleton/2.0.4/skeleton.min.css">
|
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/skeleton/2.0.4/skeleton.min.css">
|
||||||
<link rel="stylesheet" href="/css/gargoyle.css">
|
<link rel="stylesheet" href="/css/gargoyle.css">
|
||||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.6.1/jquery.min.js"></script>
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.6.1/jquery.min.js"></script>
|
||||||
<script src="/js/drake.js"></script>
|
<script src="/js/nechryael.min.js"></script>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<!-- global navigation -->
|
<!-- global navigation -->
|
||||||
@ -24,6 +24,12 @@
|
|||||||
</div>
|
</div>
|
||||||
</nav>
|
</nav>
|
||||||
|
|
||||||
|
{% if flash != null %}
|
||||||
|
<div class="flash-message {{ flash.type }}">
|
||||||
|
<p>{{ flash.msg }}</p>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
<!-- main content -->
|
<!-- main content -->
|
||||||
<div id="main-content" class="container fluid">
|
<div id="main-content" class="container fluid">
|
||||||
{% block content %}{% endblock %}
|
{% block content %}{% endblock %}
|
||||||
|
Reference in New Issue
Block a user