Server now works and compiles

This commit is contained in:
2018-07-01 11:17:29 +10:00
parent 05f3e4a51e
commit 7ef7ee6d35
7 changed files with 195 additions and 21 deletions

View File

@ -24,6 +24,19 @@
}, },
"homepage": "https://github.com/YourWishes/domsPlaceNew#readme", "homepage": "https://github.com/YourWishes/domsPlaceNew#readme",
"dependencies": { "dependencies": {
"babel-core": "^6.26.3",
"babel-loader": "^7.1.4",
"babel-polyfill": "^6.26.0",
"babel-preset-env": "^1.7.0",
"babel-preset-react": "^6.24.1",
"body-parser": "^1.18.3",
"compression-webpack-plugin": "^1.1.11",
"css-loader": "^0.28.11",
"express": "^4.16.3",
"file-loader": "^1.1.11",
"html-webpack-plugin": "^3.2.0",
"mini-css-extract-plugin": "^0.4.1",
"node-sass": "^4.9.0",
"pg-promise": "^8.4.4", "pg-promise": "^8.4.4",
"react": "^16.4.0", "react": "^16.4.0",
"react-dom": "^16.4.0", "react-dom": "^16.4.0",
@ -33,23 +46,15 @@
"react-router-dom": "^4.3.1", "react-router-dom": "^4.3.1",
"react-tap-event-plugin": "^3.0.3", "react-tap-event-plugin": "^3.0.3",
"react-transition-group": "^2.3.1", "react-transition-group": "^2.3.1",
"redux": "^4.0.0" "redux": "^4.0.0",
"sass-loader": "^7.0.3",
"style-loader": "^0.21.0",
"uglifyjs-webpack-plugin": "^1.2.7",
"url-loader": "^1.0.1",
"webpack": "^4.14.0"
}, },
"devDependencies": { "devDependencies": {
"babel-core": "^6.26.3", "react-hot-loader": "^4.3.3",
"babel-loader": "^7.1.4", "webpack-dev-server": "^3.1.4"
"babel-polyfill": "^6.26.0",
"babel-preset-env": "^1.6.1",
"babel-preset-react": "^6.24.1",
"css-loader": "^0.28.11",
"file-loader": "^1.1.11",
"html-webpack-plugin": "^3.2.0",
"node-sass": "^4.9.0",
"react-hot-loader": "^4.1.2",
"sass-loader": "^7.0.1",
"style-loader": "^0.21.0",
"url-loader": "^1.0.1",
"webpack": "^3.11.0",
"webpack-dev-server": "^2.9.1"
} }
} }

View File

@ -21,12 +21,17 @@
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//Imports
const const
path = require('path'),
ConfigurationManager = require('./../configuration/ConfigurationManager'), ConfigurationManager = require('./../configuration/ConfigurationManager'),
DatabaseConnection = require('./../database/DatabaseConnection'), DatabaseConnection = require('./../database/DatabaseConnection'),
Server = require('./../server/Server') Server = require('./../server/Server')
; ;
//Constants
const PUBLIC_PATH = path.join(__dirname, '..', '..', 'dist');
class App { class App {
constructor() { constructor() {
@ -34,6 +39,7 @@ class App {
getConfig() { return this.config; } getConfig() { return this.config; }
getDatabase() { return this.db; } getDatabase() { return this.db; }
getPublicDirectory() { return PUBLIC_PATH; }
//Primary Functions //Primary Functions
async start() { async start() {

View File

@ -27,9 +27,15 @@ const
https = require('https'), https = require('https'),
express = require('express'), express = require('express'),
bodyParser = require('body-parser'), bodyParser = require('body-parser'),
fs = require('fs') fs = require('fs'),
path = require('path'),
webpack = require('webpack'),
CompilerOptions = require('./WebpackCompilerOptions')
; ;
//Constants
const LANDING_FILE = 'index.html';
class Server { class Server {
constructor(app) { constructor(app) {
this.app = app; this.app = app;
@ -66,6 +72,7 @@ class Server {
throw new Error("Invalid SSL Cert in Server Configuration"); throw new Error("Invalid SSL Cert in Server Configuration");
} }
//TODO: Clean this up, don't use static files (use path.join etc) and should these be flat files?
let keyFile = __dirname+'./../'+this.config.ssl.key; let keyFile = __dirname+'./../'+this.config.ssl.key;
let certFile = __dirname+'./../'+this.config.ssl.cert; let certFile = __dirname+'./../'+this.config.ssl.cert;
if(!fs.existsSync(keyFile)) { if(!fs.existsSync(keyFile)) {
@ -89,6 +96,10 @@ class Server {
this.express.use(bodyParser.urlencoded({ this.express.use(bodyParser.urlencoded({
extended: true extended: true
})); }));
this.express.use(express.static('./dist'));
//Finally our catcher for all other enquiries
this.express.get('*', this.onGetRequest.bind(this));
//Create our HTTP and (if needed HTTPS) server(s) //Create our HTTP and (if needed HTTPS) server(s)
this.http = http.createServer(this.express); this.http = http.createServer(this.express);
@ -104,6 +115,9 @@ class Server {
}, this.express); }, this.express);
this.https.on('error', this.onServerError.bind(this)); this.https.on('error', this.onServerError.bind(this));
} }
//Create our bundler
this.compiler = webpack(CompilerOptions(this, this.app));
} }
getConfig() {return this.config;} getConfig() {return this.config;}
@ -114,6 +128,7 @@ class Server {
getHTTPSPort() {return this.portHTTPS;} getHTTPSPort() {return this.portHTTPS;}
getKey() {return this.key;} getKey() {return this.key;}
getCertificate() {return this.cert;} getCertificate() {return this.cert;}
getLandingFile() {return path.join(this.app.getPublicDirectory(), LANDING_FILE);}
isRunning() { isRunning() {
if(typeof this.http !== typeof undefined) { if(typeof this.http !== typeof undefined) {
@ -140,6 +155,11 @@ class Server {
port: this.port port: this.port
}; };
//Create our webpack watcher
this.watcher = this.compiler.watch({
}, this.onWatchChange.bind(this));
//Start the HTTP Server //Start the HTTP Server
this.http.listen(options, this.onServerStart.bind(this)); this.http.listen(options, this.onServerStart.bind(this));
@ -172,12 +192,18 @@ class Server {
delete this.http; delete this.http;
delete this.https; delete this.https;
delete this.stopPromise; delete this.stopPromise;
delete this.watcher;
} }
stopPromise(resolve, reject) { stopPromise(resolve, reject) {
this.stopResolve = resolve; this.stopResolve = resolve;
this.stopReject = reject; this.stopReject = reject;
if(typeof this.watcher !== typeof undefined) {
this.watcher.close(() => {
});
}
try { try {
this.http.close(this.onHTTPClosed.bind(this)); this.http.close(this.onHTTPClosed.bind(this));
} catch(e) { } catch(e) {
@ -201,6 +227,15 @@ class Server {
onHTTPSClosed() { onHTTPSClosed() {
this.resolve(); this.resolve();
} }
onWatchChange(error, stats) {
if(error) console.log(error);
}
onGetRequest(req, res) {
//Used as our "catch all get requests"
res.sendFile(this.getLandingFile());
}
} }
module.exports = Server; module.exports = Server;

View File

@ -0,0 +1,127 @@
// Copyright (c) 2018 Dominic Masters
//
// MIT License
//
// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//Includes
const
path = require('path'),
webpack = require('webpack'),
HtmlWebpackPlugin = require('html-webpack-plugin'),
CompressionPlugin = require("compression-webpack-plugin"),
UglifyJsPlugin = require('uglifyjs-webpack-plugin'),
MiniCssExtractPlugin = require("mini-css-extract-plugin")
;
//Constants
const SOURCE_DIRECTORY = './public';
const ENTRY_FILE = 'index.jsx';
const ENTRY_WRAPPER = 'index.html';
module.exports = function(server, app) {
//Create our dirs
let entryDir = path.join(app.getPublicDirectory(), '..', 'public');
//Create our output
let output = {};
//Set the entry point
output.entry = path.join(entryDir, ENTRY_FILE);
//Set the output
output.output = {
path: app.getPublicDirectory(),
filename: "app.js"
}
//Set Resolves
output.resolve = {
modules: ['node_modules', SOURCE_DIRECTORY],
extensions: ['.js', '.jsx', '.css', '.scss' ]
};
//Setup Modules
output.module = {
rules: [
{
test: /\.jsx?$/,
exclude: /node_modules/,
loaders: ['babel-loader']
},
{
test: /\.scss$|\.css$/i,
use: [
MiniCssExtractPlugin.loader,
"css-loader",
'sass-loader',
]
},
{
test: /\.jpe?g$|\.gif$|\.png$|\.svg|\.webm|\.mp4$/i,
loader: "file-loader?name=[path][name].[ext]"
},
{
test: /\.(eot|ttf|woff(2)?)(\?v=\d+\.\d+\.\d+)?/,
loader: 'url-loader'
}
]
};
//Setup the Plugins
let HTMLWebpackPluginConfig = new HtmlWebpackPlugin({
template: path.join(SOURCE_DIRECTORY, ENTRY_WRAPPER),
filename: ENTRY_WRAPPER,
inject: true
});
let UglifyPluginConfig = new UglifyJsPlugin({
test: /\.js($|\?)/i
});
let MiniCssExtractConfig = new MiniCssExtractPlugin({
filename: "[name].css",
chunkFilename: "[id].css"
})
//Set the plugins
output.plugins = [
new webpack.DefinePlugin({
'process.env.NODE_ENV': JSON.stringify('production')
}),
MiniCssExtractConfig,
HTMLWebpackPluginConfig
];
//Minimization
output.optimization = {
minimize: true,
minimizer: [
UglifyPluginConfig
]
};
//Now setup the production values
output.devtool = 'source-map';
return output;
}

View File

@ -27,7 +27,7 @@ import { connect } from 'react-redux';
import Background from './background/Background'; import Background from './background/Background';
import Header from './header/Header'; import Header from './header/Header';
import Footer from './footer/Footer'; import Footer from './footer/Footer';
import { HashRouter } from 'react-router-dom'; import { HashRouter, BrowserRouter } from 'react-router-dom';
import Routes from './page/Routes'; import Routes from './page/Routes';
class App extends React.Component { class App extends React.Component {
@ -50,12 +50,12 @@ class App extends React.Component {
if(this.props.menuOpen) clazz += " is-menu-open " if(this.props.menuOpen) clazz += " is-menu-open "
return ( return (
<HashRouter> <BrowserRouter>
<div className={clazz} ref="app"> <div className={clazz} ref="app">
<Header /> <Header />
<Routes onEntering={this.onEnteringBound} /> <Routes onEntering={this.onEnteringBound} />
</div> </div>
</HashRouter> </BrowserRouter>
); );
} }
} }

View File

@ -57,6 +57,7 @@ class ElementScrollFader extends React.Component {
} }
checkEffect() { checkEffect() {
if(typeof window === typeof undefined) return;
if(!this.refs || !this.refs.fader) return; if(!this.refs || !this.refs.fader) return;
//Get bounds //Get bounds
var rect = this.refs.fader.getBoundingClientRect(); var rect = this.refs.fader.getBoundingClientRect();

View File

@ -41,7 +41,7 @@ import Window95, {
const ExistingWorkFrame = (props) => { const ExistingWorkFrame = (props) => {
let fakeURL = props.href; let fakeURL = props.href;
if(!fakeURL.startsWith("http")) { if(!fakeURL.startsWith("http") && typeof window !== typeof undefined) {
fakeURL = window.location.protocol + fakeURL; fakeURL = window.location.protocol + fakeURL;
} }