diff --git a/.github/workflows/test.yml-template b/.github/workflows/test.yml-template
new file mode 100644
index 0000000..bb13dfc
--- /dev/null
+++ b/.github/workflows/test.yml-template
@@ -0,0 +1,23 @@
+name: Test
+
+on:
+ pull_request:
+ branches: [ master ]
+
+jobs:
+ build:
+
+ runs-on: ubuntu-latest
+
+ strategy:
+ matrix:
+ node-version: [20.x]
+
+ steps:
+ - uses: actions/checkout@v2
+ - name: Use Node.js ${{ matrix.node-version }}
+ uses: actions/setup-node@v1
+ with:
+ node-version: ${{ matrix.node-version }}
+ - run: npm install
+ - run: npm test
diff --git a/package-lock.json b/package-lock.json
index 28a4d31..f0d9216 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -11,7 +11,7 @@
"license": "GPL-3.0",
"devDependencies": {
"@mate-academy/eslint-config": "latest",
- "@mate-academy/scripts": "^1.8.6",
+ "@mate-academy/scripts": "^2.1.1",
"axios": "^1.7.2",
"eslint": "^8.57.0",
"eslint-plugin-jest": "^28.6.0",
@@ -1468,10 +1468,11 @@
}
},
"node_modules/@mate-academy/scripts": {
- "version": "1.8.6",
- "resolved": "https://registry.npmjs.org/@mate-academy/scripts/-/scripts-1.8.6.tgz",
- "integrity": "sha512-b4om/whj4G9emyi84ORE3FRZzCRwRIesr8tJHXa8EvJdOaAPDpzcJ8A0sFfMsWH9NUOVmOwkBtOXDu5eZZ00Ig==",
+ "version": "2.1.1",
+ "resolved": "https://registry.npmjs.org/@mate-academy/scripts/-/scripts-2.1.1.tgz",
+ "integrity": "sha512-Tf97p/jZ/ZRsQSPGcZf2FpvxgUCl8DiUOsiDFHj7HBN8gMK0iZOBQEtyqcFsauVUAvPP8Ayo8cAiC12MMp45iQ==",
"dev": true,
+ "license": "MIT",
"dependencies": {
"@octokit/rest": "^17.11.2",
"@types/get-port": "^4.2.0",
diff --git a/package.json b/package.json
index 8a92721..ebb46a1 100644
--- a/package.json
+++ b/package.json
@@ -17,7 +17,7 @@
"license": "GPL-3.0",
"devDependencies": {
"@mate-academy/eslint-config": "latest",
- "@mate-academy/scripts": "^1.8.6",
+ "@mate-academy/scripts": "^2.1.1",
"axios": "^1.7.2",
"eslint": "^8.57.0",
"eslint-plugin-jest": "^28.6.0",
diff --git a/public/index.html b/public/index.html
new file mode 100644
index 0000000..6b7803a
--- /dev/null
+++ b/public/index.html
@@ -0,0 +1,31 @@
+
+
+
+
+
+ Form Data
+
+
+
+ Form Data
+
+
+
diff --git a/src/createServer.js b/src/createServer.js
index 1cf1dda..dd80216 100644
--- a/src/createServer.js
+++ b/src/createServer.js
@@ -1,8 +1,113 @@
'use strict';
+const http = require('node:http');
+const fs = require('node:fs');
+const path = require('node:path');
+
function createServer() {
- /* Write your code here */
- // Return instance of http.Server class
+ const server = http.createServer((req, res) => {
+ const reqUrl = new URL(req.url || '', `http://${req.headers.host}`);
+ const pathname = reqUrl.pathname;
+
+ if (req.method === 'GET' && pathname === '/') {
+ const indexPath = path.resolve('public', 'index.html');
+
+ res.statusCode = 200;
+ res.setHeader('Content-Type', 'text/html');
+ fs.createReadStream(indexPath).pipe(res);
+
+ return;
+ }
+
+ if (req.method === 'GET' && pathname === '/add-expense') {
+ res.statusCode = 400;
+
+ return res.end('Only POST method allowed');
+ }
+
+ if (req.method === 'POST' && pathname === '/add-expense') {
+ let body = '';
+
+ req.on('data', (chunk) => {
+ body += chunk.toString();
+ });
+
+ req.on('end', () => {
+ let dataObj;
+
+ try {
+ const contentType = req.headers['content-type'];
+
+ if (contentType.includes('application/json')) {
+ dataObj = JSON.parse(body);
+ } else if (
+ contentType.includes('application/x-www-form-urlencoded')
+ ) {
+ const parsed = new URLSearchParams(body);
+
+ dataObj = Object.fromEntries(parsed.entries());
+ } else {
+ res.writeHead(400, { 'Content-Type': 'text/plain' });
+
+ return res.end('Unsupported content type');
+ }
+
+ const { date, title, amount } = dataObj;
+
+ if (!date || !title || !amount) {
+ res.writeHead(400, { 'Content-Type': 'text/plain' });
+
+ return res.end('Invalid data format');
+ }
+
+ const filePath = path.resolve('db', 'expense.json');
+
+ fs.readFile(filePath, (err, fileData) => {
+ let parsedFileData = [];
+
+ if (err) {
+ res.writeHead(500, { 'Content-Type': 'text/plain' });
+
+ return res.end(`Server error: ${err}`);
+ } else {
+ parsedFileData = JSON.parse(fileData);
+
+ if (typeof parsedFileData === 'object') {
+ if (Object.keys(parsedFileData).length) {
+ parsedFileData = [parsedFileData];
+ parsedFileData.push(dataObj);
+ } else if (!Object.keys(parsedFileData).length) {
+ parsedFileData = dataObj;
+ } else if (Array.isArray(parsedFileData)) {
+ parsedFileData.push(dataObj);
+ }
+ }
+
+ const newData = JSON.stringify(parsedFileData, null, 2);
+
+ fs.writeFile(filePath, newData, (error) => {
+ if (error) {
+ res.writeHead(500, { 'Content-Type': 'text/plain' });
+
+ return res.end(`Server error: ${error}`);
+ }
+ res.writeHead(200, { 'Content-Type': 'application/json' });
+ res.end(newData);
+ });
+ }
+ });
+ } catch (e) {
+ res.writeHead(400, { 'Content-Type': 'text/plain' });
+ res.end(`Invalid request body: ${e.message}`);
+ }
+ });
+ } else {
+ res.statusCode = 404;
+ res.end('Not found');
+ }
+ });
+
+ return server;
}
module.exports = {