Skip to content
Snippets Groups Projects
Commit 0a9623ca authored by alrwasheda's avatar alrwasheda :speech_balloon:
Browse files

Merge branch 'main' of git.imp.fu-berlin.de:swp-ws21-fahrtenbuch/team-einhorn/fahrtenbuch into main

parents 18b1072f b40c50ef
No related branches found
No related tags found
No related merge requests found
# This file is a template, and might need editing before it works on your project. image: node:lts-alpine
# To contribute improvements to CI/CD templates, please follow the Development guide at:
# https://docs.gitlab.com/ee/development/cicd/templates.html
# This specific template is located at:
# https://gitlab.com/gitlab-org/gitlab/-/blob/master/lib/gitlab/ci/templates/Nodejs.gitlab-ci.yml
# Official framework image. Look for the different tagged releases at:
# https://hub.docker.com/r/library/node/tags/
image: node:16.2.0
# Pick zero or more services to be used on all builds.
# Only needed when using a docker container to run your tests in.
# Check out: http://docs.gitlab.com/ee/ci/docker/using_docker_images.html#what-is-a-service
services: services:
- postgres:latest - postgres:latest
...@@ -25,34 +14,52 @@ variables: ...@@ -25,34 +14,52 @@ variables:
POSTGRES_USER: $POSTGRES_USER POSTGRES_USER: $POSTGRES_USER
POSTGRES_PASSWORD: $POSTGRES_PASSWORD POSTGRES_PASSWORD: $POSTGRES_PASSWORD
POSTGRES_HOST_AUTH_METHOD: trust POSTGRES_HOST_AUTH_METHOD: trust
SKIP_PREFLIGHT_CHECK: "true"
# This folder is cached between builds # These folders and files are cached between builds
# https://docs.gitlab.com/ee/ci/yaml/index.html#cache
cache: cache:
paths: paths:
- node_modules/ - client/node_modules/
- dist/server.js - server/node_modules/
- server/.env
before_script: - server/dist/
- npm install
- nohup npm run watch:server &
# Stages for better overview
stages: stages:
- prepare
- build - build
- launch
- test - test
check_build: mirror_dockerfile_server:
stage: prepare
script:
- cd server && npm install
mirror_dockerfile_client:
stage: prepare
script:
- cd client && npm install
build_server:
stage: build stage: build
script: script:
- npm run build - cd server && npm run build
#launch_build: test_server:
# stage: launch stage: test
# script: # I only run the server, since server tests are (almost) always against the API
# - nohup npm run watch:server & before_script:
- cd server && npm run start &
script:
- cd server && npm run test
test_run: test_client:
stage: test stage: test
# If specific content needs to be tested, both the server and client need to be up and running
before_script:
- cd client && npm start &
- cd server && npm start &
script: script:
- npm test - cd client && npm run test
\ No newline at end of file
\ No newline at end of file
/** @type {import('ts-jest/dist/types').InitialOptionsTsJest} */
module.exports = {
preset: 'ts-jest',
testEnvironment: 'node',
};
\ No newline at end of file
This diff is collapsed.
...@@ -9,7 +9,6 @@ ...@@ -9,7 +9,6 @@
"@testing-library/jest-dom": "^5.15.0", "@testing-library/jest-dom": "^5.15.0",
"@testing-library/react": "^11.2.7", "@testing-library/react": "^11.2.7",
"@testing-library/user-event": "^12.8.3", "@testing-library/user-event": "^12.8.3",
"@types/jest": "^26.0.24",
"@types/node": "^12.20.36", "@types/node": "^12.20.36",
"@types/react": "^17.0.34", "@types/react": "^17.0.34",
"@types/react-dom": "^17.0.11", "@types/react-dom": "^17.0.11",
...@@ -28,7 +27,7 @@ ...@@ -28,7 +27,7 @@
"scripts": { "scripts": {
"start": "react-scripts start", "start": "react-scripts start",
"build": "react-scripts build", "build": "react-scripts build",
"test": "react-scripts test", "test": "jest",
"eject": "react-scripts eject" "eject": "react-scripts eject"
}, },
"eslintConfig": { "eslintConfig": {
...@@ -49,5 +48,11 @@ ...@@ -49,5 +48,11 @@
"last 1 safari version" "last 1 safari version"
] ]
}, },
"proxy": "http://server:4000" "proxy": "http://server:4000",
"devDependencies": {
"@types/supertest": "^2.0.11",
"jest": "^27.3.1",
"supertest": "^6.1.6",
"ts-jest": "^27.0.7"
}
} }
require('dotenv').config(); import request from 'supertest';
const request = require('supertest');
describe('Serving Pages', () => {
describe('GET /', () => { /*
it('responds with successful (2XX) request and html file', (done) => { * NOTICE:
* Those are structural test examples.
* I've made it, that they all expect 404 errors,
* since the database models have changes while the API hasn't
*/
it('should serve index page', (done) => {
request("http://localhost:3000") request("http://localhost:3000")
.get('/') .get('/')
.set('Accept', 'text/html') .set('Accept', 'text/html')
.expect('Content-Type', /html/) .expect('Content-Type', /html/)
.expect(200, done); .expect(200, done);
}); }, 60000);
}); });
\ No newline at end of file
...@@ -21,6 +21,7 @@ ...@@ -21,6 +21,7 @@
"jsx": "react-jsx" "jsx": "react-jsx"
}, },
"include": [ "include": [
"src" "src",
"tests"
] ]
} }
# Testing
## Running Tests
Running tests is as simple as going either into the `client` or `server` directory and running `npm test`. Jest will then automatically take all the `*.test.ts` files and run them
> Tip: Before running any test against the client or server, make sure you have the server and/or client running
It's very important to note that the Pipeline on Gitlab will execute all the tests, so expect failed pipelines if you push broken code.
## Writing tests
To create your own tests, just create a new file in the `test` directory of the part you want to test. You could also change an existing file or even create a subdirectory with files. It's important to note that any and all tests files should end with `.test.ts` otherwise Jest will not run them.
### Structuring tests
Since testing is done with [jest](https://www.npmjs.com/package/jest) we can follow a specific structure like this:
```javascript
describe('Test', () => {
it('should do something', (done) => {
// Do something
});
it('should do something else', (done) => {
// Do something else
});
});
```
The `describe` function can be used to group specific tests into one Test Suite, so that you can have similar tests be run together. In this example the Suite would be calles "Test". `it` is used to define a single test. The first argument is simply the test name, the second parameter will be the actual test thats run. Whatever you decide to define here, will be executed. The `done` parameter is optional and can be used if you expect callback.
> Tip: make sure to create tests that maybe only test a single function or a small piece of code and be sure to use descriptive test names, since tests can already serve as documentation.
## API Testing
Using [supertest](https://www.npmjs.com/package/supertest) and the structure above, we can easily create API tests. We only have to modify our structure a little bit.
```javascript
import request from 'supertest';
describe('Test', () => {
it('should do something', (done) => {
//The URL should be always this with our project settings
request("http://localhost:4000")
.get('/api/v1/test')
.set('Accept', 'application/json')
.expect('Content-Type', /json/)
.expect(200, done)
});
});
```
This example sends a GET request to the `/api/v1/test` endpoint and expects to receive a 200 status code and a JSON response.
### GET Requests
GET Requests are already demonstrated in the example above.
>Tip: if you need to send a GET request with authentication, you can use the `.auth(username, password)` function to do so.
### POST Requests
As you can guess the requests are the same as GET requests, but with the exception of the `.post()` method.
```javascript
import request from 'supertest';
describe('Test', () => {
it('should do something', (done) => {
//The URL should be always this with our project settings
request("http://localhost:4000")
.post('/api/v1/users')
.send({name:'john'})
.set('Accept', 'application/json')
.expect('Content-Type', /json/)
.expect(200, done)
});
});
```
### Expectations
If you expect your request to yield a response with specific content, you can use the `.expect()` function. For example if you post a Username and expect to get a fixed ID, you can do this:
```javascript
describe('POST /user', () => {
it('user.name should be an case-insensitive match for "john"', (done) => {
request(app)
.post('/user')
.send('name=john') // x-www-form-urlencoded upload
.set('Accept', 'application/json')
.expect(function(res) {
res.body.id = 'some fixed id';
res.body.name = res.body.name.toLowerCase();
})
.expect(200, {
id: 'some fixed id',
name: 'john'
}, done);
});
});
```
You can do a lot more with supertest, basically everything you can to with superagent, so refer to the [supertest documentation](https://www.npmjs.com/package/supertest) and the [superagent documentation](https://www.npmjs.com/package/superagent)for more information.
## React Testing
Testing React/the Frontend works similar as API testing.
Refer to the official [Jest Testing Documentation](https://jestjs.io/docs/tutorial-react) to find out how to do it
...@@ -5,3 +5,9 @@ DB_HOST=localhost ...@@ -5,3 +5,9 @@ DB_HOST=localhost
DB_DRIVER=postgres DB_DRIVER=postgres
DB_PASSWORD=postgres DB_PASSWORD=postgres
TEST_DB_NAME=test TEST_DB_NAME=test
EMAIL_SERVER=smtp.example.com
EMAIL_IS_TLS=false
EMAIL_USERNAME=username
EMAIL_PASSWORD=pw
EMAIL_FROM_ADDR=username@example.com
Application_Name=Fahrtenbuch Wassersportzentrum
/** @type {import('ts-jest/dist/types').InitialOptionsTsJest} */
module.exports = {
preset: 'ts-jest',
testEnvironment: 'node',
};
\ No newline at end of file
This diff is collapsed.
...@@ -8,22 +8,30 @@ ...@@ -8,22 +8,30 @@
"build": "tsc", "build": "tsc",
"server": "nodemon server --ignore client", "server": "nodemon server --ignore client",
"start": "node ./dist/server.js", "start": "node ./dist/server.js",
"test": "echo \"Error: no test specified\" && exit 1" "test": "jest"
}, },
"author": "", "author": "",
"license": "ISC", "license": "ISC",
"dependencies": { "dependencies": {
"dotenv": "^10.0.0", "dotenv": "^10.0.0",
"express": "^4.17.1", "express": "^4.17.1",
"handlebars": "^4.7.7",
"nodemailer": "^6.7.2",
"pg": "^8.7.1", "pg": "^8.7.1",
"sequelize": "^6.9.0" "sequelize": "^6.9.0"
}, },
"devDependencies": { "devDependencies": {
"@types/nodemailer": "^6.4.4",
"@types/express": "^4.17.13", "@types/express": "^4.17.13",
"@types/jest": "^27.0.3",
"@types/node": "^16.11.6", "@types/node": "^16.11.6",
"@types/pg": "^8.6.1", "@types/pg": "^8.6.1",
"@types/supertest": "^2.0.11",
"@types/validator": "^13.6.6", "@types/validator": "^13.6.6",
"jest": "^27.3.1",
"nodemon": "^2.0.14", "nodemon": "^2.0.14",
"supertest": "^6.1.6",
"ts-jest": "^27.0.7",
"ts-node": "^10.4.0", "ts-node": "^10.4.0",
"typescript": "^4.4.4" "typescript": "^4.4.4"
} }
......
<!DOCTYPE html>
<html lang="en" style="background:#f5f5f5" xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
</head>
<body style="margin: 0; padding: 0; font-family:Lato, sans-serif;">
<table border="0" cellpadding="0" cellspacing="0" width="100%">
<tr>
<td style="padding: 20px 0 30px 0;">
<table align="center" border="0" cellpadding="0" cellspacing="0" width="100%">
<tr>
<td bgcolor="#6b9e1f" height="65" style="color: white; font-family: Arial, sans-serif; font-size: 26px; padding: 10px;">
{{application_name}} Benachrichtigung
</td>
</tr>
<tr>
<td bgcolor="#ffffff" style="padding: 10px;">
<span>Es wurde eine Anmerkung zu <a style="letter-spacing: 1px; color: #37557D;" href="#">{{boat}}</a> verfasst:</span>
<table role="presentation" bgcolor="#EAF0F6" width="80%" align="center" style="margin-top: 10px">
<tr>
<td align="left" style="padding: 30px 30px;">
{{notification}}
</td>
</tr>
</table>
<table role="presentation" bgcolor="#37557D" style="color: white" width="80%" align="center">
<tr>
<td align="left" style="padding: 5px 30px;">
Author: {{author}}
</td>
<td align="right" style="padding: 5px 30px;">
Time: {{timestamp}}
</td>
</tr>
</table>
</td>
</tr>
</table>
<!-- Manage Notifications -->
<table bgcolor="#F5F8FA" width="100%" >
<tr>
<td align="left" style="padding: 20px 30px;">
<a style="font-size: 9px; text-transform:uppercase; letter-spacing: 1px; color: #CBD6E2;" href="{{manage_notification_link}}"> Manage Notifications </a>
</td>
</tr>
</table>
</td>
</tr>
</table>
</body>
</html>
{application_name}} Benachrichtigung
Es wurde eine Anmerkung zu {{boat}} verfasst:
{{notification}}
Author: {{author}} Time: {{timestamp}}
import nodemailer from "nodemailer";
import path from 'path';
import fs from 'fs';
import Handlebars from 'handlebars';
let transporter: nodemailer.Transporter = undefined;
let ready = false;
let create = async () => {
transporter = nodemailer.createTransport({
host: process.env.EMAIL_SERVER,
port: process.env.EMAIL_Port ? Number(process.env.EMAIL_Port) : (Boolean(process.env.EMAIL_IS_TLS) ? 465 : 587),
secure: Boolean(process.env.EMAIL_IS_TLS),
auth: {
user: process.env.EMAIL_USERNAME,
pass: process.env.EMAIL_PASSWORD
}
});
transporter.verify(function (error, success) {
if (error) {
console.log("Error verifying Email connection: " + error);
process.exit(9)
} else {
ready = true;
}
});
}
create();
// Construct Templates and load to RAM
let templating: { [templateName: string]: { html: HandlebarsTemplateDelegate, txt: HandlebarsTemplateDelegate } } = {};
fs.readdirSync('./assets').forEach(file => {
if (file.includes('.html')) {
try {
templating[file.split('.')[0]] = {
html: Handlebars.compile(fs.readFileSync(path.join("./assets", file), 'utf8')),
txt: Handlebars.compile(fs.readFileSync(path.join("./assets", file.split('.')[0] + '.txt'), 'utf8'))
};
} catch (e) {
console.log(`Couldn't load template '${file}'`, e)
}
}
});
const sendMail = (template: "notification", to: string, subject: string, replace: { [toReplace: string]: string }) => {
if (!ready || !templating[template]) {
console.log(ready, templating[template])
return
}
transporter.sendMail({
from: `"${process.env.Application_Name}" <${process.env.EMAIL_FROM_ADDR}>`,
to: to,
subject: subject,
text: templating[template].txt(replace),
html: templating[template].html(replace)
}, (err, info) => {
if (err) return console.log('error', JSON.stringify(err));
//if (info) return logger.log('info', JSON.stringify(info));
return null;
});
}
export default sendMail
import "dotenv/config"; import "dotenv/config";
import express from "express"; import express from "express";
// import connectAndInit from "./db/db_init";
import showAllDBs from "./db/showAllDBs"; import showAllDBs from "./db/showAllDBs";
import showTables from "./db/showDBTables"; import showTables from "./db/showDBTables";
import initializeDatabase from './db'; import initializeDatabase from "./db";
// import apiRouter from "./routes";
import sendMail from "./mail";
let init = async () => { let init = async () => {
const sequelize = await initializeDatabase(); const sequelize = await initializeDatabase();
......
require('dotenv').config();
import request from 'supertest';
describe('GET Endpoints', () => {
/*
* NOTICE:
* Those are structural test examples.
* I've made it, that they all expect 404 errors,
* since the database models have changes while the API hasn't
*/
it('if response is 404, server is alive (since GET / isn\'t defined)', (done) => {
request("http://localhost:4000")
.get('/')
.set('Accept', 'text/html')
.expect('Content-Type', /html/)
.expect(404, done);
});
it('Get Vehicle Type with ID (returns 404 since no data)', (done) => {
request("http://localhost:4000")
.get('/vehicleType/1')
.set('Accept', 'text/html')
.expect('Content-Type', /html/)
.expect(404, done);
});
it('Get all Vehicle Types', (done) => {
request("http://localhost:4000")
.get('/vehicleType')
.set('Accept', 'text/html')
.expect('Content-Type', /html/)
.expect(404, done);
});
});
\ No newline at end of file
...@@ -15,6 +15,7 @@ ...@@ -15,6 +15,7 @@
} }
}, },
"include": [ "include": [
"src/**/*" "src/**/*",
"tests/*"
] ]
} }
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment