Selve applikasjonen kan du finne her: https://rombestilling.vercel.app/
Prosjektet er et bookingsystem for rom. Her kan brukere reservere ulike rom og underdeler av rom. Reservasjon kan skje som enkeltperson eller på vegne av en gruppe. Rom og brukere opprettes og administreres av administratorer som har full tilgang til hele systemet. Administratorer kan batch-opprette mange bruker samtidig ved hjelp av CSV-filer, hvorpå de nye brukerne vil motta en epost med link til side der passord kan settes. Swagger-docs finnes på /swagger-ui/
- Applikasjonen
- Databaseskjema
- Teknologier
- Applikasjonstruktur
- Sikkerhet
- API-dokumentasjon
- Installasjon
- CI/CD
- Medlemmer
Prosjektet består av en backend del og en frontend del hvorav begge delene er utviklet med moderne og stabile rammeverk. På frontend benytter vi oss av React, som er blant det mest brukte rammeverket idag. Vi brukte Typescript for å skrive frontend da det sørger for statisk typing, noe som betyr letter feilsøking og mindre feil. På backend har vi benyttet oss Spring Boot som rammeverk for å bygge REST-apiet vårt. Dette har vi gjort sammen med å bruke Kotlin og Gradle. Kotlin er et moderne språk som lar deg skrive både funksjonell og objekt orientert kode. Vi har benyttet oss av både CI og CD under prosjektet for å sørge for den beste kontinuerlige utviklingen. Vi har benyttet oss av Azure for å deploye backend hver gang vi oppdaterer main branchen vår, noe som har gjort at vi til enhver tid kan teste den nyeste funksjonaliteten vår, live. For å deploye frontend bruker vi Vercel, slik at den nyeste backenden alltid er i samspill med den nyeste frontenden.
Applikasjonen har to viktige aktører:
- Endebruker
- Adminbruker
En endebruker har begrenset tilgang og kan
- Logge inn
- Tilbakestille passord
- Endre på profil
- Liste tilgjengelige rom
- Reservere rom (på vegne av seg selv eller en gruppe)
En adminbruker har tilgang til alle aspekter en endebruker har, i tillegg til følgende:
- Lage endebrukere (evt ved csv-fil)
- Gjøre endebrukere til adminbrukere
- Administrere brukere
- Administrere rom
- Administrere grupper
- Administrere gruppemedlemskap (evt opprette ved csv-fil)
- Se statistikk for rom
Når adminbrukere oppretter vanlige brukere, enten enkeltbrukere eller ved csv, vil de nye brukere få tilsendt en epost med en link til å tilbakestille passord. De kan da trykke på denne og velge et passord for kontoen sin og logge inn i applikasjonen.
Nye brukeres gyldighet blir satt til ett år. Når denne går ut på dato vil endebrukeren miste bruker-rollen og dermed misten tilgang til systemet. Dette gjøres via en planlagt oppgave (scheduled task/cron-job) som kjører hver dag kl 22:00.
Følgende er vårt databaseskjema:
Her er det verdt å nevne litt om måten vi implementerte seksjoner/rom (sections). Seksjoner er strukturert som et tre hvor en rot-node tilsvarer et rom, og en barnenode er underseksjoner. Barnenodene vil her være løvnoder, da treet kun kan ha 2 nivåer. Dette gjør det enkelt å utvide til flere underseksjoner av underseksjoner og kunne reservere både rom og underseksjoner.
- React - Javascript-rammeverk for å bygge brukergrensesnitt
- Spring Boot - Server-side rammeverk
- Docker - Rammeverk for konteinerisering
- MySQL - Database (DBMS)
- H2 Database Engine - In-memory database til automatisert testing
- Swagger - API dokumentasjon
- Material UI - UI tema/design
- JWT - Autentiseringsmekanisme for REST APIer
- QueryDSL - Spørrespråk for å konstruere SQL-spørringer
- opencsv - CSV-analyserer for Java
- awaitility - Java DSL for å synkronisere asynkrone operasjoner
Vi har forsøkt å legge opp strukturen på en domenesentrert måte, med pakker for hver av de tilknyttede domenemodellene våre.
Hver pakke er deretter delt opp i underpakker som reflekterer de ulike elementene som utgjør en naturlig del av applikasjonen.
Controller-klassene er view-laget og tilbyr REST-endepunkter for klienter over HTTP.
Service-klassene inneholder det meste av forretningslogikken slik at den er uavhengig av laget over.
Repositories er våre repository-klasser og er ansvarlig for å kommunisere med databasen.
Modellene våre ligger i model
pakken og deres respektive DTOer (Data Transfer Objects) ligger i dto
. Dette er gjort for å decouple modell-laget fra view-laget slik at de også kan utvikle seg uavhengig av hverandre.
Sikkerhet vært et gjennomgående fokus punkt under hele prosjektet da vi mener at dette er at vi de viktigste fokusene man må ivareta som utvikler. APIet vårt er designet rundt REST prinsipper, noe som gjør at vi derfor benytter oss av stateless authentication. Dette gjør vi ved hjelp av Json Web Tokens. En gyldig innlogget bruker vil til enhver tid ha en Acess token og en refresh token tilgjengelig. Acess token gjør at brukeren kan få tilgang på innhold, men denne har en gyldighet på 15 minutter, av sikkerhetsmessig årsaker. Bruker vil da automatisk benytte seg av refresh token for å få en ny og oppdatert acess token. Alt dette skjer automatisk, slik at bruker ikke skal trenge å bruke energi på det, samtidig som det gir brukeren en sikker opplevelse.
Vi har også implementert en enkel form for Refresh Token Rotation (RTR): Et refresh token kan bare brukes én gang og hvis en prøver å gjenbruke en refresh token vil alle refresh tokens utgitt til brukeren etter denne bli ugyldige.
Vi har også hatt fokus på Zero Trust konseptet, og har derfor kun åpnet et fåtall endepunkter, mens resten krever autentisjon. Alle endepunkter krever authentisering og adminrolle. Vi har som nevnt åpnet et fåtall endepunkter til brukere med brukerrollen, slik at man må være registrert som en gyldig bruker med denne rollen.
I tilfellet en breach skulle skje så har vi lagret alle passordene med Bcrypt algoritmen kombinert med salt. Vi har også utivklet et eget autorisasjons system, for å validere et brukere har rettighetene de trenger for å kunne utføre en handling. Dette er for å forhindre at brukere kan endre eller slette andre brukeres innhold.
Dette prosjektet bruker Swagger 3 til API-dokumentasjon. Denne finnes på http://localhost:8080/swagger-ui (https://rombestilling.azurewebsites.net/swagger-ui/)
For å autentisere kan "Authentication Services"-specet brukes:
Her kan man prøve ut forespørselen og legge inn epost og passord:
Da vil man få tilbake et access token under feltet "token" som man bruker til å logge inn:
Legg så tokenet inn som authentication, husk å inkludere prefikset "Bearer":
Backend
Frontend
- Node:
git clone https://github.com/olros/IDATT2105-project.git
cd IDATT2105-project/backend/
# Med GNU Make:
make run-unix
# Uten GNU Make:
./gradlew jibDockerBuild --image=rombestilling.azurecr.io/rombestilling:latest
docker-compose -f docker-compose.azure.yml up --build
# Utvikling:
make db / docker-compose up
./gradlew bootRun
# Med GNU Make:
make run-windows
# Uten GNU Make:
gradlew jibDockerBuild --image=rombestilling.azurecr.io/rombestilling:latest
docker-compose -f docker-compose.azure.yml up --build
# Utvikling:
make db / docker-compose up
gradlew bootRun
Applikasjonen benytter seg av epost for å sende ut informasjon til nye brukere og mer. For å sette opp epost må du lage en fil som heter emailconfig.properties
under backend/src/main/resources/mail/
. Under ser du feltene som du må fylle ut for å sette opp epost med en gmail konto.
spring.mail.host=smtp.gmail.com
spring.mail.port=587
spring.mail.username=EPOST
spring.mail.password=PASSORD
NB!: Det kan hende du må endre på instillinger inne på selve epost kontoen din for å tillate å sende mail på denne måten.
cd IDATT2105-project/frontend/
# Installer avhengigheter
yarn
# Kjør i utviklingsmodus
yarn start
Når applikasjonene har startet opp vil applikasjonen være tilgjengelig på http://localhost:3000/ og REST API-et vil da være tilgjengelig på http://localhost:8080/
Det ligger en adminbruker og en vanlig bruker inne i systemet man kan bruke under testing med følgende innlogging:
Admin:
- Epost: admin@test.com
- Passord: admin
Vanlig bruker:
- Epost: user@test.com
- Passord: user
Disse brukerne er lagt inn kun for bruk under utvikling og skal på ingen måte brukes i et virkelig scenario.
Tester har blitt kjørt fortløpende med CI for både frontend og backend gjennom Github Actions.
CI runs backend:
CI runs frontend:
Fortløpende deployment er løst gjennom Azure for backend og Vercel for frontend.
I Azure har vi opprettet et privat Container Registry som vi kan pushe container images til. Det gjør vi gjennom en CD-workflow der vi bruker jib, et verktøy laget av Google for å lage container images, som et Gradle plugin til å lage et Docker-image av applikasjonen og så pushe det til registry'et. I Azure har vi også satt opp en Azure App Service-instans som kjører en Docker-Compose fil som kjører applikasjonen og en MySQL-database. Ettersom databasen ligger i App Servicen blir den tømt hver gang applikasjonen starter på nytt, noe er praktisk under utvikling men naturligvis ville vært endret i et virkelig scenario. Det er også satt opp en Webhook som automatisk restarter App Service med nytt image fra Container Registry ved ny push til registry'et.
Vercel har et veldig enkelt oppsett for frontend applikasjoner. Det eneste som har vært nødvendig fra vår side er å opprette et nytt prosjekt i Vercel og knytte det til dette Git-repositoryet. Etter det har Vercel automatisk oppdatert nettsiden ved push til main, samt laget såkalte preview deployments til hver Pull Request som enkelt gjør oss i stand til å se hvordan nettsiden ser ut i nye branches.
Hermann Owren Elton -
Mads Lundegaard -
Olaf Rosendahl -
Eirik Steira -