diff --git a/.env b/.env new file mode 100644 index 0000000..843e7fa --- /dev/null +++ b/.env @@ -0,0 +1,9 @@ +PORT=80 +POSTGRES_USER=admin +POSTGRES_PASSWORD=admin +KEYCLOAK_USER=admin +KEYCLOAK_PASSWORD=admin +KEYCLOAK_FRONTEND_URL=http://auth.localhost +KEYCLOAK_REALM=master +KEYCLOAK_FRONTEND_CLIENT=frontend +KEYCLOAK_BACKEND_CLIENT=backend \ No newline at end of file diff --git a/.gitignore b/.gitignore index 234ad04..0c41bd0 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,5 @@ frontend/dist +postgres **/node_modules **/__pycache__ diff --git a/README.md b/README.md index 57cdcf4..423d94f 100644 --- a/README.md +++ b/README.md @@ -13,4 +13,41 @@ Features: * Editing records (including date) * Display records in cronological order with pagination * Search by records description -* Watch stats in superset (calories chart) \ No newline at end of file +* Watch stats in superset (calories chart) +* Docker compose + +API docs: + +Record { + id: int + description: string + weight: number + calories: number + calspergram: number + date: datetime +} + +GET /records?offset=0&amount=5 +[ + { + date: string + records: Record[] + }[5] +] + +POST /records +{ + description: string + weight: number + calories: number + calspergram: number +} -> Record + +UPDATE /records +{ + description?: string + weight?: number + calories?: number + calspergram?: number + date?: datetime +} -> Record \ No newline at end of file diff --git a/compose.yml b/compose.yml index df55f1a..44ae975 100644 --- a/compose.yml +++ b/compose.yml @@ -1,45 +1,43 @@ services: - frontend: - build: - context: ./frontend + nginx: + image: nginx:latest + volumes: + - ./nginx:/etc/nginx/conf.d + - ./frontend/dist:/usr/share/nginx/html ports: - - 3000:80 + - 80:80 + depends_on: + - backend + - keycloak backend: build: context: ./backend - ports: - - 9000:9000 + depends_on: + - keycloak keycloak: image: quay.io/keycloak/keycloak:latest - container_name: keycloak environment: - KC_DB=postgres - - KC_DB_URL=jdbc:postgresql://db:5432/keycloak - - KC_DB_USERNAME=keycloak - - KC_DB_PASSWORD=keycloakpassword - - KEYCLOAK_ADMIN=admin - - KEYCLOAK_ADMIN_PASSWORD=adminpassword - - KEYCLOAK_FRONTEND_URL=http://localhost/auth - ports: - - "8080:8080" - command: start-dev + - KC_DB_URL=jdbc:postgresql://postgres:5432/keycloak + - KC_DB_USERNAME=${POSTGRES_USER} + - KC_DB_PASSWORD=${POSTGRES_PASSWORD} + - KEYCLOAK_ADMIN=${KEYCLOAK_USER} + - KEYCLOAK_ADMIN_PASSWORD=${KEYCLOAK_PASSWORD} + - KEYCLOAK_FRONTEND_URL=${KEYCLOAK_FRONTEND_URL} + volumes: + - ./keycloak:/opt/keycloak/data/import + command: > + start-dev --import-realm depends_on: - - db + - postgres - db: - image: postgres:15 - container_name: keycloak-db + postgres: + image: postgres:latest environment: POSTGRES_DB: keycloak - POSTGRES_USER: keycloak - POSTGRES_PASSWORD: keycloakpassword + POSTGRES_USER: ${POSTGRES_USER} + POSTGRES_PASSWORD: ${POSTGRES_PASSWORD} volumes: - - keycloak-data:/var/lib/postgresql/data - ports: - - "5432:5432" - -volumes: - keycloak-data: - \ No newline at end of file + - ./postgres/data:/var/lib/postgresql/data diff --git a/frontend/Dockerfile b/frontend/Dockerfile index e5d3571..7065402 100644 --- a/frontend/Dockerfile +++ b/frontend/Dockerfile @@ -1,12 +1,5 @@ -FROM node:18 as build +FROM node as build +COPY . /app WORKDIR /app -COPY package*.json ./ RUN npm install -COPY . . -RUN npx webpack build -FROM nginx:1.25 -RUN rm /etc/nginx/conf.d/default.conf -COPY nginx.conf /etc/nginx/conf.d/default.conf -COPY --from=build /app/dist /usr/share/nginx/html -EXPOSE 80 -CMD ["nginx", "-g", "daemon off;"] +CMD npm run watch diff --git a/frontend/package-lock.json b/frontend/package-lock.json index 7641ad4..826270c 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -10,6 +10,7 @@ "license": "ISC", "dependencies": { "@react-keycloak/web": "^3.4.0", + "dotenv": "^16.4.7", "keycloak-js": "^26.1.0", "react": "^19.0.0", "react-dom": "^19.0.0" @@ -1561,6 +1562,18 @@ "tslib": "^2.0.3" } }, + "node_modules/dotenv": { + "version": "16.4.7", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.7.tgz", + "integrity": "sha512-47qPchRCykZC03FhkYAhrvwU4xDBFIj1QPqaarj6mdM/hgUzfPHcpkHJOn3mJAufFeeAxAzeGsr5X0M4k6fLZQ==", + "license": "BSD-2-Clause", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://dotenvx.com" + } + }, "node_modules/dunder-proto": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", diff --git a/frontend/package.json b/frontend/package.json index 4501b8b..002696e 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -3,7 +3,7 @@ "version": "1.0.0", "main": "index.js", "scripts": { - "test": "echo \"Error: no test specified\" && exit 1" + "watch": "npx webpack --watch" }, "keywords": [], "author": "", @@ -11,6 +11,7 @@ "description": "", "dependencies": { "@react-keycloak/web": "^3.4.0", + "dotenv": "^16.4.7", "keycloak-js": "^26.1.0", "react": "^19.0.0", "react-dom": "^19.0.0" @@ -26,4 +27,4 @@ "webpack-cli": "^6.0.1", "webpack-dev-server": "^5.2.0" } -} \ No newline at end of file +} diff --git a/frontend/src/App.tsx b/frontend/src/App.tsx index a5bd525..bbd0084 100644 --- a/frontend/src/App.tsx +++ b/frontend/src/App.tsx @@ -1,62 +1,14 @@ import React, { useEffect } from 'react'; - -import Keycloak from 'keycloak-js'; - -const keycloak = new Keycloak({ - url: 'http://localhost:8080/', - realm: 'master', - clientId: 'react-client', -}); - -// let kc = new Keycloak({ -// url: 'http://localhost:8080/', -// realm: 'master', -// clientId: 'react-client', -// }); - -// kc.init({ -// onLoad: 'check-sso', -// silentCheckSsoRedirectUri: window.location.origin + "/silent-check-sso.html", -// checkLoginIframe: false, -// pkceMethod: 'S256' -// }).then((auth) => { -// if (!auth) { -// window.location.reload(); -// } else { -// console.info("Authenticated"); -// console.log('auth', auth) -// console.log('Keycloak', kc) -// kc.onTokenExpired = () => { -// console.log('token expired') -// } -// } -// }, () => { -// console.error("Authenticated Failed"); -// }); - -// console.log(kc.responseType) +import keycloak from './keycloak'; const App: React.FC = () => { - useEffect(() => { - try { - keycloak.init({ - onLoad: 'login-required', - checkLoginIframe: false, - pkceMethod: 'S256' - }).then(auth => { - if (auth) { - console.log(auth); - console.log('User is authenticated'); - } else { - console.log('User is not authenticated'); - } - }); - } catch (error) { - console.error('Failed to initialize adapter:', error); - } - }, []); - - return