diff --git a/.gitignore b/.gitignore index 5a977d4..9cfc02d 100644 --- a/.gitignore +++ b/.gitignore @@ -62,4 +62,5 @@ typings/ /package-lock.json /nbproject/private/ .vscode -.serverless \ No newline at end of file +.serverless +.cache \ No newline at end of file diff --git a/src/private/package.json b/src/private/package.json index eb9cc3c..f438fa1 100644 --- a/src/private/package.json +++ b/src/private/package.json @@ -9,7 +9,7 @@ "start:dev": "cross-env NODE_ENV=development \"serverless offline --port 3001\"", "start": "npm run start:prod", "watch": "npm run start:dev", - "deploy": "echo \"Building private\"", + "deploy": "serverless deploy", "test": "yarn jest" }, "repository": { @@ -34,7 +34,7 @@ "serverless-plugin-include-dependencies": "^3.2.1" }, "devDependencies": { - "@types/jest": "^24.0.18", + "@types/jest": "^25.1.1", "@types/node": "^12.7.1", "@types/nodemailer": "^6.2.1", "cross-env": "^5.2.0", diff --git a/src/private/serverless.yml b/src/private/serverless.yml index b7e747d..31bfe21 100644 --- a/src/private/serverless.yml +++ b/src/private/serverless.yml @@ -26,13 +26,19 @@ provider: functions: ping: - handler: dist/index.ping + handler: dist/functions/ping/ping.ping events: - - http: ANY ping + - http: + method: GET + path: ping + cors: true sendMail: handler: dist/functions/mail/send.sendMail events: - - http: ANY mail/send + - http: + method: POST + path: mail/send + cors: true plugins: - serverless-plugin-include-dependencies diff --git a/src/private/src/functions/mail/send.ts b/src/private/src/functions/mail/send.ts index 60e98c1..745ecb5 100644 --- a/src/private/src/functions/mail/send.ts +++ b/src/private/src/functions/mail/send.ts @@ -10,16 +10,13 @@ export interface sendMailParams { } export const sendMail = withHandler(async (e,c) => { - //Required - if(!e || !e.body) return { statusCode: 400, body: 'Missing Contact Details' }; + 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' }; @@ -55,5 +52,5 @@ export const sendMail = withHandler(async (e,c) => { Time: ${new Date().toLocaleString()} ` }); - return { statusCode: 200, body: true } + return { statusCode: 200, body: x && x.accepted && x.accepted.length } }); \ No newline at end of file diff --git a/src/private/src/functions/ping/__tests__/ping.test.ts b/src/private/src/functions/ping/__tests__/ping.test.ts index 8cd6318..e143b7a 100644 --- a/src/private/src/functions/ping/__tests__/ping.test.ts +++ b/src/private/src/functions/ping/__tests__/ping.test.ts @@ -2,6 +2,7 @@ import { ping } from './../ping'; describe('ping', () => { it('shold return a hello world and a 200 code', async () => { - await expect(ping()).resolves.toStrictEqual({ statusCode: 200, body: 'Thank funk!' }); + await expect(ping()).resolves.toHaveProperty('body', 'Thank funk!'); + await expect(ping()).resolves.toHaveProperty('statusCode', 200); }) }) \ No newline at end of file diff --git a/src/private/src/handler/__tests__/handler.test.ts b/src/private/src/handler/__tests__/handler.test.ts index 7186dd5..a2fcd41 100644 --- a/src/private/src/handler/__tests__/handler.test.ts +++ b/src/private/src/handler/__tests__/handler.test.ts @@ -1,4 +1,4 @@ -import { withHandler, APICallable } from './../handler'; +import { withHandler, APICallable, DEFAULT_HEADERS } from './../handler'; describe('withHandler', () => { it('should wrap an async function into a serverless callbackable function', () => { @@ -28,7 +28,7 @@ describe('withHandler', () => { await new Promise(resolve => setImmediate(resolve)); expect(fn).toHaveBeenCalledWith(null, { body: '"Hello World"', statusCode: 200, - headers: { 'Content-Type': 'application/json'} + headers: DEFAULT_HEADERS }); }); diff --git a/src/private/src/handler/handler.ts b/src/private/src/handler/handler.ts index 19ceb1b..99757fa 100644 --- a/src/private/src/handler/handler.ts +++ b/src/private/src/handler/handler.ts @@ -1,3 +1,9 @@ +export const DEFAULT_HEADERS = { + 'Content-Type': 'application/json', + 'Access-Control-Allow-Origin': '*', + 'Access-Control-Allow-Credentials': true +} + export type APIResponse = { statusCode?:number; body:any; @@ -28,42 +34,36 @@ export interface APIEvent { export const withHandler = (callable:APICallable) => { return (event?:APIEvent, context?, callback?) => { - if(event && event.headers && event.headers['Content-Type']) { - let contentType = event.headers['Content-Type']; - if(contentType.indexOf('application/json') !== -1) { - try { - let body:T = JSON.parse(event.body as any); - event.body = body; - } catch(e) { - callback(null, { - statusCode: 400, - headers: { 'Content-Type': 'application/json' }, - body: JSON.stringify('Invalid body') - }) - } - } + try { + if(!event) throw new Error(); + if(event.body) event.body = JSON.parse(event.body as any) as T; + } catch(e) { + console.error(e); + callback(null, { + statusCode: 400, headers: DEFAULT_HEADERS, + body: JSON.stringify('Invalid body') + }); } return callable(event, context).then(d => { - if(callback) { - let contentType = (d.headers ? d.headers['Content-Type']:null) ||'application/json'; - let json = contentType == 'application/json'; + if(!callback) return d; + let contentType = (d.headers?d.headers['Content-Type']:null) ||'application/json'; + let json = contentType.includes('application/json'); - callback(null, { - ...d, - body: json ? JSON.stringify(d.body) : d.body, - statusCode: d.statusCode || 200, - headers: { - ...(d.headers||{}), - "Content-Type": contentType - } - }); - } + callback(null, { + ...d, + body: json ? JSON.stringify(d.body) : d.body, + statusCode: d.statusCode || 200, + headers: { + ...DEFAULT_HEADERS, + ...(d.headers||{}) + } + }); return d; }).catch(ex => { if(callback) { - callback(null, { statusCode: 500, body: null, }); + callback(null, { statusCode: 500, body: null, headers: DEFAULT_HEADERS }); } throw ex; }) diff --git a/src/private/src/index.ts b/src/private/src/index.ts deleted file mode 100644 index e69de29..0000000 diff --git a/src/public/package.json b/src/public/package.json index 63641e6..fdadf92 100644 --- a/src/public/package.json +++ b/src/public/package.json @@ -7,7 +7,7 @@ "develop": "gatsby develop", "start": "npm run develop", "serve": "gatsby serve", - "deploy": "echo \"Building public\"", + "deploy": "serverless client deploy", "clean": "gatsby clean" }, "repository": { @@ -27,6 +27,7 @@ }, "homepage": "https://domsplace.com", "dependencies": { + "axios": "^0.19.2", "babel-plugin-styled-components": "^1.10.7", "gatsby": "^2.18.12", "gatsby-image": "^2.2.39", @@ -47,11 +48,14 @@ "yup": "^0.28.1" }, "devDependencies": { + "serverless": "^1.63.0", + "serverless-finch": "^2.5.2", + "serverless-plugin-include-dependencies": "^4.0.1", "@types/node": "^13.5.0", "@types/react": "^16.9.19", "@types/react-dom": "^16.9.5", - "@types/styled-components": "^4.4.2", "@types/react-helmet": "^5.0.15", + "@types/styled-components": "^4.4.2", "@types/yup": "^0.26.29" }, "browserslist": [ diff --git a/src/public/serverless.yml b/src/public/serverless.yml index 9bca3ad..6552523 100644 --- a/src/public/serverless.yml +++ b/src/public/serverless.yml @@ -3,6 +3,13 @@ service: domsplace frameworkVersion: ">=1.26.0" +provider: + name: aws + runtime: nodejs10.x + stage: ${opt:stage, "test"} + region: ap-southeast-2 + memorySize: 512 + package: excludeDevDependencies: false individually: true @@ -16,6 +23,6 @@ plugins: custom: client: bucketName: domsplace-${self:provider.stage}-${self:provider.region}-public - distributionFolder: frontned/public/ + distributionFolder: public/ indexDocument: index.html errorDocument: index.html \ No newline at end of file diff --git a/src/public/src/api/Client.tsx b/src/public/src/api/Client.tsx new file mode 100644 index 0000000..0f7d81c --- /dev/null +++ b/src/public/src/api/Client.tsx @@ -0,0 +1,5 @@ +import axios from 'axios'; + +export const Client = axios.create({ + baseURL: 'https://api.domsplace.com/v1/' +}); diff --git a/src/public/src/api/SendMail.tsx b/src/public/src/api/SendMail.tsx index 86a7475..117dfbf 100644 --- a/src/public/src/api/SendMail.tsx +++ b/src/public/src/api/SendMail.tsx @@ -1,12 +1,8 @@ -export const sendMail = (name:string, email:string, message:string) => { - return fetch('https://api.domsplace.com/v1/mail/send', { - method: 'POST', - body: JSON.stringify({ - name, email, message - }), - headers: { - 'Content-Type': 'application/json', - 'Accept': 'application/json', - } - }).then(d => d.json()); -} \ No newline at end of file +import { Client } from "./Client"; + +export const sendMail = (name:string, email:string, message:string) => Client.post('mail/send', { + name, email, message +}); + +///@ts-ginore +(globalThis as any)['sendMail' as any] = sendMail as any; \ No newline at end of file