diff --git a/.gitignore b/.gitignore
index 2f7ad6c..9cfc02d 100644
--- a/.gitignore
+++ b/.gitignore
@@ -34,6 +34,8 @@ build/Release
# Dependency directories
node_modules/
+node_modules_pc/
+node_modules_surface/
jspm_packages/
# Typescript v1 declaration files
@@ -57,8 +59,8 @@ typings/
# dotenv environment variables file
.env
-dist/
/package-lock.json
-/dist
-src/private/data
-/nbproject/private/
\ No newline at end of file
+/nbproject/private/
+.vscode
+.serverless
+.cache
\ No newline at end of file
diff --git a/.travis.yml b/.travis.yml
new file mode 100644
index 0000000..9afebea
--- /dev/null
+++ b/.travis.yml
@@ -0,0 +1,32 @@
+language: node_js
+
+node_js:
+ - "10.16.3"
+
+cache:
+ yarn: true
+ directories:
+ - node_modules
+
+matrix:
+ include:
+ - env: PROJECT=src/private/
+ include:
+ - env: PROJECT=src/public/
+
+install:
+ - cd $PROJECT
+ - yarn global add serverless
+ - yarn install
+
+script:
+ - yarn test
+ - yarn build
+
+deploy:
+ provider: script
+ script:
+ - yarn deploy
+ skip_cleanup: true
+ on:
+ branch: master
\ No newline at end of file
diff --git a/LICENSE b/LICENSE
deleted file mode 100644
index bb9d3e8..0000000
--- a/LICENSE
+++ /dev/null
@@ -1,21 +0,0 @@
-MIT License
-
-Copyright (c) 2018 Dominic Masters
-
-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.
diff --git a/README.md b/README.md
deleted file mode 100644
index 12b98dd..0000000
--- a/README.md
+++ /dev/null
@@ -1,25 +0,0 @@
-
-
domsPlace();
-
-My personal website, written in Javascript, HTML, CSS using Node, React, SCSS, Webpack and Babel.
-
-Yet another redesign after many, the site is permanently stuck in limbo as I don't have as much time to work on it as I would like.
-
-## Roadmap
-Plans to add are:
-
- - ~~Favicon~~
- - Short Blog Page
- - Featured Video (Code is ready, video needs to be made)
- - More Social integration and show off my social pages
- - Faster loading (SVG Backgrounds are quite large)
- - ~~Responsive Image Loading~~
- - ~~Async Image/Video loading for content (no point just yet)~~
- - ~~Async page offsetting,~~ as well as proper loading templates
- - Improved SEO
- - Reduce Divitis
- - Restore previously removed page transitions
- - ~~Convert some of the SVGs into responsive PNGs~~ Unsure if I'll stick with this permanently
- - Work on Async Sections
- - Proper Server
- - Adjust the order import order so to help CSS Overrides.
diff --git a/package.json b/package.json
deleted file mode 100644
index 44eb900..0000000
--- a/package.json
+++ /dev/null
@@ -1,45 +0,0 @@
-{
- "name": "domsplace",
- "version": "6.0.1",
- "description": "Personal website for Dominic \"YouWish\" Masters.",
- "main": "dist/private/",
- "scripts": {
- "test": "jest",
- "start": "node dist/private/",
- "build": "tsc -p . && webpack -p --env.production",
- "watch": "cross-env NODE_ENV=DEVELOPMENT npm run start",
- "heroku-postbuild": "tsc -p . && webpack -p --env.production"
- },
- "repository": {
- "type": "git",
- "url": "git+https://github.com/YourWishes/domsPlaceNew.git"
- },
- "keywords": [
- "domsplace",
- "personal",
- "portfolio",
- "website"
- ],
- "author": "Dominic Masters",
- "license": "MIT",
- "bugs": {
- "url": "https://github.com/YourWishes/domsPlaceNew/issues"
- },
- "homepage": "https://github.com/YourWishes/domsPlaceNew#readme",
- "dependencies": {
- "@types/animejs": "^2.0.2",
- "@yourwishes/app-email": "^1.0.3",
- "@yourwishes/app-simple-react": "^2.8.7",
- "animejs": "^3.0.1",
- "react-helmet": "^5.2.1"
- },
- "devDependencies": {
- "@types/jest": "^24.0.15",
- "cross-env": "^5.2.0",
- "jest": "^24.1.0",
- "typescript": "^3.5.3",
- "webpack-cli": "^3.3.5",
- "webpack-dev-middleware": "^3.6.1",
- "webpack-hot-middleware": "^2.24.3"
- }
-}
diff --git a/src/private/.gitignore b/src/private/.gitignore
new file mode 100644
index 0000000..f770a93
--- /dev/null
+++ b/src/private/.gitignore
@@ -0,0 +1,3 @@
+dist/
+*.log
+*.lock
\ No newline at end of file
diff --git a/src/private/api/contact/index.ts b/src/private/api/contact/index.ts
deleted file mode 100644
index 4bf68ce..0000000
--- a/src/private/api/contact/index.ts
+++ /dev/null
@@ -1,78 +0,0 @@
-// Copyright (c) 2019 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, sublicen se, 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.
-
-import { RESPONSE_OK, RESPONSE_BAD_REQUEST, RESPONSE_INTERNAL_ERROR } from '@yourwishes/app-api';
-import { ServerAPIRequest, ServerAPIResponse, ServerAPIHandler } from '@yourwishes/app-server';
-import { isValidEmail } from '@yourwishes/app-email';
-import { domsPlaceApp } from './../../app/';
-
-export const EMAIL_MAXLENGTH = 256;
-export const NAME_MAXLENGTH = 128;
-export const MESSAGE_MAXLENGTH = 8192;
-
-export class sendContact extends ServerAPIHandler {
- constructor() {
- super('POST', '/contact/send');
- }
-
- async onRequest(request:ServerAPIRequest):Promise {
- if(!request.hasString('email', EMAIL_MAXLENGTH)) return { code: RESPONSE_BAD_REQUEST, data: 'Missing or invalid email' };
- if(!request.hasString('name', NAME_MAXLENGTH)) return { code: RESPONSE_BAD_REQUEST, data: 'Missing or invalid name' };
- if(!request.hasString('message', MESSAGE_MAXLENGTH)) return { code: RESPONSE_BAD_REQUEST, data: 'Missing or invalid message' };
-
- let email = request.getString('email', EMAIL_MAXLENGTH);
- if(!isValidEmail(email)) return { code: RESPONSE_BAD_REQUEST, data: 'Missing or invalid email' };
-
- let name = request.getString('name', NAME_MAXLENGTH);
- let message = request.getString('message', MESSAGE_MAXLENGTH);
-
- //Prepare to send email
- let app:domsPlaceApp = request.owner.app as domsPlaceApp;
-
- //First send an email to the site owner
- request.owner.logger.debug(`Sending email from ${email}...`);
- let ownRes = await app.email.sendMail(app.domsPlace.contact, 'Contact Message Received', `
- Contact Message received from ${email}, who wrote:
- ${message}
- `, `
-
- `);
- if(!ownRes) return { code: RESPONSE_INTERNAL_ERROR, data: false };
-
- //Now Send an email to the client
- let clientRes = await app.email.sendMail(app.domsPlace.contact, 'Contact Message Sent', `
- Contact Message Sent! Thanks for reaching out,
- if this was not you then ignore this email.
- `, `
-
Contact Message Sent! Thanks for reaching out. If this was not you then ignore this email.
- `);
- if(!clientRes) return { code: RESPONSE_INTERNAL_ERROR, data: false };
- request.owner.logger.debug(`...Done`);
-
- //OK!
- return { code: RESPONSE_OK, data: true };
- }
-}
diff --git a/src/private/api/index.ts b/src/private/api/index.ts
deleted file mode 100644
index 371663a..0000000
--- a/src/private/api/index.ts
+++ /dev/null
@@ -1,24 +0,0 @@
-// Copyright (c) 2019 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.
-
-export * from './contact/';
diff --git a/src/private/app/index.ts b/src/private/app/index.ts
deleted file mode 100644
index 37992ce..0000000
--- a/src/private/app/index.ts
+++ /dev/null
@@ -1,57 +0,0 @@
-// Copyright (c) 2019 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.
-
-import { App } from '@yourwishes/app-base';
-import { ServerModule } from '@yourwishes/app-server';
-import { ReactModule } from '@yourwishes/app-react';
-import { ISimpleReactApp, SimpleReactModule } from '@yourwishes/app-simple-react';
-import { IEmailApp, EmailModule } from '@yourwishes/app-email';
-import { domsPlaceModule } from './../module/';
-
-import { domsPlaceCompiler } from './../compiler/';
-
-export class domsPlaceApp extends App implements ISimpleReactApp, IEmailApp {
- server:ServerModule;
- email:EmailModule;
- react:ReactModule;
- simpleReact:SimpleReactModule;
- domsPlace:domsPlaceModule;
-
- constructor() {
- super();
-
- this.server = new ServerModule(this);
- this.email = new EmailModule(this);
- this.react = new ReactModule(this);
- this.simpleReact = new SimpleReactModule(this);
- this.domsPlace = new domsPlaceModule(this);
-
- [
- this.server, this.email, this.react, this.simpleReact, this.domsPlace
- ].forEach(e => this.modules.addModule(e));
- }
-
- getCompiler() {
- return new domsPlaceCompiler();
- }
-}
diff --git a/src/private/compiler/index.ts b/src/private/compiler/index.ts
deleted file mode 100644
index 67a2a89..0000000
--- a/src/private/compiler/index.ts
+++ /dev/null
@@ -1,50 +0,0 @@
-// Copyright (c) 2019 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.
-
-import { SimpleReactCompiler } from '@yourwishes/app-simple-react';
-
-const gtag = `
-`;
-
-const content = `
- Please Wait...
-`;
-
-export class domsPlaceCompiler extends SimpleReactCompiler {
- constructor() {
- super({
- title: 'domsPlace - Personal Site of Dominic Masters',
- keywords: 'domsplace, programming, gaming, shopify, livestreaming, dominic, masters, dom',
- description: 'domsPlace is the home of programmer and developer Dominic Masters, specialising in eCommerce and full-stack development solutions for a wide range of platforms, primarily Shopify.',
- app_handle: 'domsPlace',
- language: 'en-AU',
- gtag,
- page_content: content
- });
- }
-}
diff --git a/src/private/index.ts b/src/private/index.ts
deleted file mode 100644
index 84a2ad8..0000000
--- a/src/private/index.ts
+++ /dev/null
@@ -1,28 +0,0 @@
-// Copyright (c) 2019 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.
-
-import { domsPlaceApp } from './app/';
-
-const app = new domsPlaceApp();
-
-app.init().catch(e => app.logger.error(e));
diff --git a/src/private/jest.config.js b/src/private/jest.config.js
new file mode 100644
index 0000000..af3e56e
--- /dev/null
+++ b/src/private/jest.config.js
@@ -0,0 +1,14 @@
+module.exports = {
+ "roots": [
+ "/src"
+ ],
+ "transform": {
+ "^.+\\.ts?$": "ts-jest"
+ },
+ "testRegex": "(/__tests__/.*|(\\.|/)(test|spec))\\.ts?$",
+ "moduleFileExtensions": [
+ "ts",
+ "js",
+ "json"
+ ]
+}
diff --git a/src/private/module/index.ts b/src/private/module/index.ts
deleted file mode 100644
index 5f988a8..0000000
--- a/src/private/module/index.ts
+++ /dev/null
@@ -1,51 +0,0 @@
-// Copyright (c) 2019 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.
-
-import { Module } from '@yourwishes/app-base';
-import { domsPlaceApp } from './../app/';
-import { sendContact } from './../api/';
-
-export const CONFIG_CONTACT = 'domsplace.contact';
-
-export class domsPlaceModule extends Module {
- app:domsPlaceApp;
- contact:string;
-
- constructor(app:domsPlaceApp) {
- super(app);
-
- app.server.api.addAPIHandler(new sendContact());
- }
-
- loadPackage() {return require('./../../../package.json');}
-
- async init(): Promise {
- let { config } = this.app;
- if(!config.has(CONFIG_CONTACT)) throw new Error("Missing contact reciving details from configuration!");
-
- this.contact = config.get(CONFIG_CONTACT);
- }
-
- async destroy(): Promise {
- }
-}
diff --git a/src/private/package.json b/src/private/package.json
new file mode 100644
index 0000000..f438fa1
--- /dev/null
+++ b/src/private/package.json
@@ -0,0 +1,51 @@
+{
+ "name": "domsplace-backend",
+ "version": "7.0.0",
+ "description": "Personal website for Dominic \"YouWish\" Masters.",
+ "main": "./dist/private/",
+ "scripts": {
+ "build": "tsc -p .",
+ "start:prod": "cross-env NODE_ENV=production \"serverless offline\"",
+ "start:dev": "cross-env NODE_ENV=development \"serverless offline --port 3001\"",
+ "start": "npm run start:prod",
+ "watch": "npm run start:dev",
+ "deploy": "serverless deploy",
+ "test": "yarn jest"
+ },
+ "repository": {
+ "type": "git",
+ "url": "git+https://github.com/YourWishes/domsPlace.git"
+ },
+ "keywords": [
+ "domsplace",
+ "personal",
+ "portfolio",
+ "website"
+ ],
+ "author": "Dominic Masters",
+ "license": "MIT",
+ "bugs": {
+ "url": "https://github.com/YourWishes/domsPlace/issues"
+ },
+ "homepage": "https://github.com/YourWishes/domsPlace#readme",
+ "dependencies": {
+ "email-validator": "^2.0.4",
+ "nodemailer": "^6.3.0",
+ "serverless-plugin-include-dependencies": "^3.2.1"
+ },
+ "devDependencies": {
+ "@types/jest": "^25.1.1",
+ "@types/node": "^12.7.1",
+ "@types/nodemailer": "^6.2.1",
+ "cross-env": "^5.2.0",
+ "escape-html": "^1.0.3",
+ "jest": "^24.9.0",
+ "serverless": "^1.51.0",
+ "serverless-finch": "^2.4.3",
+ "serverless-offline": "^5.11.0",
+ "ts-jest": "^24.1.0",
+ "ts-node": "^8.3.0",
+ "typescript": "^3.5.3",
+ "utility-types": "^3.7.0"
+ }
+}
diff --git a/src/private/serverless.yml b/src/private/serverless.yml
new file mode 100644
index 0000000..31bfe21
--- /dev/null
+++ b/src/private/serverless.yml
@@ -0,0 +1,53 @@
+org: yourwishes
+service: domsplace
+
+frameworkVersion: ">=1.26.0"
+
+package:
+ excludeDevDependencies: false
+ individually: true
+ include:
+ - dist/**
+
+provider:
+ name: aws
+ runtime: nodejs10.x
+ stage: ${opt:stage, "prod"}
+ region: ap-southeast-2
+ memorySize: 512
+ deploymentBucket:
+ name: domsplace-${self:provider.stage}-${self:provider.region}-private
+ environment:
+ EMAIL_HOST: ${self:custom.variables.email.host}
+ EMAIL_PORT: ${self:custom.variables.email.port}
+ EMAIL_USER: ${self:custom.variables.email.user}
+ EMAIL_PASS: ${self:custom.variables.email.pass}
+ EMAIL_DEST: ${self:custom.variables.email.dest}
+
+functions:
+ ping:
+ handler: dist/functions/ping/ping.ping
+ events:
+ - http:
+ method: GET
+ path: ping
+ cors: true
+ sendMail:
+ handler: dist/functions/mail/send.sendMail
+ events:
+ - http:
+ method: POST
+ path: mail/send
+ cors: true
+
+plugins:
+ - serverless-plugin-include-dependencies
+ - serverless-offline
+
+custom:
+ ssm: '/aws/reference/secretsmanager/prod.domsPlace.'
+ serverless-offline:
+ disableCookieValidation: true
+ port: 3001
+ variables:
+ email: ${ssm:${self:custom.ssm}email~true}
\ No newline at end of file
diff --git a/src/private/src/functions/mail/__tests__/send.test.ts b/src/private/src/functions/mail/__tests__/send.test.ts
new file mode 100644
index 0000000..37a2a39
--- /dev/null
+++ b/src/private/src/functions/mail/__tests__/send.test.ts
@@ -0,0 +1,8 @@
+import { sendMail } from './../send';
+
+describe('sendMail', () => {
+ it('should require a body', async () => {
+ await expect(sendMail({ body: null })).resolves.toHaveProperty('statusCode', 400);
+ });
+
+})
\ No newline at end of file
diff --git a/src/private/src/functions/mail/send.ts b/src/private/src/functions/mail/send.ts
new file mode 100644
index 0000000..0b83717
--- /dev/null
+++ b/src/private/src/functions/mail/send.ts
@@ -0,0 +1,56 @@
+import { withHandler } from "../../handler/handler";
+import { validate } from 'email-validator';
+import { createTransport } from 'nodemailer';
+import * as escapeHTML from 'escape-html';
+
+export interface sendMailParams {
+ name:string;
+ email:string;
+ message:string;
+}
+
+export const sendMail = withHandler(async (e,c) => {
+ if(!e.body) return { statusCode: 400, body: 'Missing Contact Details' };
+
+ let { name, email, message } = e.body;
+ if(!name) return { statusCode: 400, body: 'Missing Contact Name' };
+ if(!email) return { statusCode: 400, body: 'Missing Contact Email' };
+ if(!message) return { statusCode: 400, body: 'Missing Contact Message' };
+
+ //Validate
+ if(name.length > 128 || !name.replace(/\s/g, '').length) return { statusCode: 400, body: 'Invalid Name' };
+ if(!validate(email)) return { statusCode: 400, body: 'Invalid Email' };
+ if(message.length > 10000 || !message.replace(/\s/g,'').length) {
+ return { statusCode: 400, body: 'Invalid Messatge' };
+ }
+
+ //Prepare mail
+ let {
+ EMAIL_HOST, EMAIL_PORT, EMAIL_USER, EMAIL_PASS, EMAIL_DEST, EMAIL_FROM
+ } = process.env
+
+ let transporter = createTransport({
+ host: EMAIL_HOST,
+ port: parseInt(EMAIL_PORT),
+ secure: true,
+ auth: {
+ user: EMAIL_USER, pass: EMAIL_PASS
+ }
+ });
+
+ let x = await transporter.sendMail({
+ from: `${name} <${email}>`,
+ to: EMAIL_DEST,
+ subject: 'Contact Message Received',
+ text: `Contact Message Received:\n${message}\nFrom: ${name} ${email}`,
+ html: `
+
Contact Message Received
+
You have received a contact message from ${escapeHTML(name)} - ${escapeHTML(email)} who wrote: