From f6a000b42a03296292de33527f9e31a0560b5440 Mon Sep 17 00:00:00 2001 From: Eden Xu <130119691+FlyingPufferFish@users.noreply.github.com> Date: Mon, 7 Oct 2024 22:48:40 +1100 Subject: [PATCH 1/4] build(database): modify db generation to be in `update` mode --- backend/src/main/resources/application.properties | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/backend/src/main/resources/application.properties b/backend/src/main/resources/application.properties index cf27fd9..33e9ae2 100644 --- a/backend/src/main/resources/application.properties +++ b/backend/src/main/resources/application.properties @@ -15,7 +15,7 @@ quarkus.datasource.password = ${QUARKUS_DATASOURCE_PASSWORD} quarkus.datasource.jdbc.url = ${QUARKUS_DATASOURCE_JDBC_URL} # drop and create the database at startup (use `update` to only update the schema) -quarkus.hibernate-orm.database.generation = create-drop +quarkus.hibernate-orm.database.generation = update # transaction timeout quarkus.transaction-manager.default-transaction-timeout=320s From f8e7d4e8fc6d19deaf705031c42bc95add298e4c Mon Sep 17 00:00:00 2001 From: tungkhanhh Date: Tue, 8 Oct 2024 00:27:05 +1100 Subject: [PATCH 2/4] refactor: :recycle: refactor files, bring to "rest" package, reformat code --- .../src/main/java/org/acme/domain/Room.java | 27 ++++++++----------- .../main/java/org/acme/domain/Student.java | 24 +++++++---------- .../main/java/org/acme/domain/Timetable.java | 27 +++++++------------ .../src/main/java/org/acme/domain/Unit.java | 21 +++++---------- .../acme/{domain => rest}/RoomResource.java | 13 ++++----- .../{domain => rest}/StudentResource.java | 13 ++++----- .../acme/{ => rest}/TimetableResource.java | 21 ++++++--------- .../acme/{domain => rest}/UnitResource.java | 13 ++++----- .../java/org/acme/security/jpa/Startup.java | 1 - .../main/java/org/acme/security/jpa/User.java | 3 ++- .../TimetableConstraintProviderTest.java | 16 +++++------ 11 files changed, 69 insertions(+), 110 deletions(-) rename backend/src/main/java/org/acme/{domain => rest}/RoomResource.java (78%) rename backend/src/main/java/org/acme/{domain => rest}/StudentResource.java (76%) rename backend/src/main/java/org/acme/{ => rest}/TimetableResource.java (95%) rename backend/src/main/java/org/acme/{domain => rest}/UnitResource.java (78%) diff --git a/backend/src/main/java/org/acme/domain/Room.java b/backend/src/main/java/org/acme/domain/Room.java index bd0997b..1d063ee 100644 --- a/backend/src/main/java/org/acme/domain/Room.java +++ b/backend/src/main/java/org/acme/domain/Room.java @@ -1,18 +1,13 @@ package org.acme.domain; -import java.util.ArrayList; -import java.util.List; - +import ai.timefold.solver.core.api.domain.lookup.PlanningId; import com.fasterxml.jackson.annotation.JsonIgnore; import com.fasterxml.jackson.annotation.JsonIgnoreProperties; - -import ai.timefold.solver.core.api.domain.lookup.PlanningId; import io.quarkus.hibernate.orm.panache.PanacheEntity; -import jakarta.persistence.CascadeType; -import jakarta.persistence.Entity; -import jakarta.persistence.FetchType; -import jakarta.persistence.ManyToMany; -import jakarta.persistence.OneToMany; +import jakarta.persistence.*; + +import java.util.ArrayList; +import java.util.List; /** * Represents a room. @@ -21,7 +16,7 @@ */ @Entity public class Room extends PanacheEntity { - + @PlanningId public String roomCode; @@ -55,11 +50,11 @@ public Room() { /** * Creates a room with its ID and capacity. * - * @param id The room’s id. - * @param buildingId The building that the room belongs to. - * @param campus The campus that the room belongs to. - * @param capacity The room's capacity. - * @param isLab Whether the room is a laboratory. + * @param id The room’s id. + * @param buildingId The building that the room belongs to. + * @param campus The campus that the room belongs to. + * @param capacity The room's capacity. + * @param isLab Whether the room is a laboratory. */ public Room(String id, String buildingId, String campus, int capacity, boolean isLab) { this.roomCode = id; diff --git a/backend/src/main/java/org/acme/domain/Student.java b/backend/src/main/java/org/acme/domain/Student.java index 614513a..7804ba7 100644 --- a/backend/src/main/java/org/acme/domain/Student.java +++ b/backend/src/main/java/org/acme/domain/Student.java @@ -1,19 +1,13 @@ package org.acme.domain; -import java.util.ArrayList; -import java.util.List; -import java.util.Objects; - import com.fasterxml.jackson.annotation.JsonIgnore; import com.fasterxml.jackson.annotation.JsonIgnoreProperties; - import io.quarkus.hibernate.orm.panache.PanacheEntity; -import jakarta.persistence.CascadeType; -import jakarta.persistence.Entity; -import jakarta.persistence.FetchType; -import jakarta.persistence.JoinColumn; -import jakarta.persistence.JoinTable; -import jakarta.persistence.ManyToMany; +import jakarta.persistence.*; + +import java.util.ArrayList; +import java.util.List; +import java.util.Objects; /** * Represents a student. @@ -21,16 +15,16 @@ * @author Jet Edge */ @Entity -public class Student extends PanacheEntity{ +public class Student extends PanacheEntity { public String name; @JsonIgnoreProperties("students") @ManyToMany(fetch = FetchType.LAZY, cascade = CascadeType.ALL) @JoinTable( - name = "student_unit", - joinColumns = @JoinColumn(name = "student_id"), - inverseJoinColumns = @JoinColumn(name = "unit_id") + name = "student_unit", + joinColumns = @JoinColumn(name = "student_id"), + inverseJoinColumns = @JoinColumn(name = "unit_id") ) @JsonIgnore public List units = new ArrayList(); diff --git a/backend/src/main/java/org/acme/domain/Timetable.java b/backend/src/main/java/org/acme/domain/Timetable.java index 07a139d..325b0ea 100644 --- a/backend/src/main/java/org/acme/domain/Timetable.java +++ b/backend/src/main/java/org/acme/domain/Timetable.java @@ -6,23 +6,16 @@ import ai.timefold.solver.core.api.domain.solution.ProblemFactCollectionProperty; import ai.timefold.solver.core.api.domain.valuerange.ValueRangeProvider; import ai.timefold.solver.core.api.score.buildin.hardsoft.HardSoftScore; +import com.fasterxml.jackson.annotation.JsonIgnoreProperties; +import com.fasterxml.jackson.annotation.JsonProperty; import io.quarkus.hibernate.orm.panache.PanacheEntity; -import jakarta.persistence.CascadeType; -import jakarta.persistence.ElementCollection; -import jakarta.persistence.Entity; -import jakarta.persistence.FetchType; -import jakarta.persistence.JoinColumn; -import jakarta.persistence.JoinTable; -import jakarta.persistence.ManyToMany; +import jakarta.persistence.*; import java.time.DayOfWeek; import java.time.LocalTime; import java.util.ArrayList; import java.util.List; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; -import com.fasterxml.jackson.annotation.JsonProperty; - /** * Represents a timetable, the solution from the program. * @@ -44,16 +37,16 @@ public class Timetable extends PanacheEntity { /* * Rooms can belong to multiple timetables because timetables are generated - * on a per-campus basis, and although each room can only belong to one + * on a per-campus basis, and although each room can only belong to one * campus, the user may choose to generate multiple timetables for each * campus, hence the many-to-many relationship */ @JsonIgnoreProperties("timetables") @ManyToMany(fetch = FetchType.LAZY, cascade = CascadeType.ALL) @JoinTable( - name = "room_timetable", - joinColumns = @JoinColumn(name = "timetable_id"), - inverseJoinColumns = @JoinColumn(name = "room_id") + name = "room_timetable", + joinColumns = @JoinColumn(name = "timetable_id"), + inverseJoinColumns = @JoinColumn(name = "room_id") ) @ProblemFactCollectionProperty @JsonProperty(access = JsonProperty.Access.WRITE_ONLY) @@ -68,9 +61,9 @@ public class Timetable extends PanacheEntity { @JsonIgnoreProperties("timetables") @ManyToMany(fetch = FetchType.LAZY, cascade = CascadeType.ALL) @JoinTable( - name = "unit_timetable", - joinColumns = @JoinColumn(name = "timetable_id"), - inverseJoinColumns = @JoinColumn(name = "unit_id") + name = "unit_timetable", + joinColumns = @JoinColumn(name = "timetable_id"), + inverseJoinColumns = @JoinColumn(name = "unit_id") ) @PlanningEntityCollectionProperty public List units; diff --git a/backend/src/main/java/org/acme/domain/Unit.java b/backend/src/main/java/org/acme/domain/Unit.java index 8bc6d00..eb4cef6 100644 --- a/backend/src/main/java/org/acme/domain/Unit.java +++ b/backend/src/main/java/org/acme/domain/Unit.java @@ -3,14 +3,10 @@ import ai.timefold.solver.core.api.domain.entity.PlanningEntity; import ai.timefold.solver.core.api.domain.lookup.PlanningId; import ai.timefold.solver.core.api.domain.variable.PlanningVariable; +import com.fasterxml.jackson.annotation.JsonIgnore; +import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import io.quarkus.hibernate.orm.panache.PanacheEntity; -import jakarta.persistence.CascadeType; -import jakarta.persistence.Entity; -import jakarta.persistence.FetchType; -import jakarta.persistence.JoinColumn; -import jakarta.persistence.ManyToMany; -import jakarta.persistence.ManyToOne; -import jakarta.persistence.Transient; +import jakarta.persistence.*; import java.time.DayOfWeek; import java.time.Duration; @@ -18,9 +14,6 @@ import java.util.ArrayList; import java.util.List; -import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonIgnoreProperties; - /** * Represents a unit. * @@ -55,8 +48,8 @@ public class Unit extends PanacheEntity { /* * currently each unit only has 1 'slot' on the timetable, so it can only - * be associated with one room, but in the final product, we would most - * likely have to change this to a many-to-many relationship + * be associated with one room, but in the final product, we would most + * likely have to change this to a many-to-many relationship * i.e. list of Rooms, because we might want to separate lecture/tutorial * etc. */ @@ -128,7 +121,7 @@ public Unit(int unitID, String name, String course, Duration duration, List students, boolean wantsLab, Room room) { this.unitId = unitID; @@ -194,7 +187,7 @@ public List getStudents() { public void setStudents(List students) { this.students = students; - this.setStudentsUnits();; + this.setStudentsUnits(); } /** diff --git a/backend/src/main/java/org/acme/domain/RoomResource.java b/backend/src/main/java/org/acme/rest/RoomResource.java similarity index 78% rename from backend/src/main/java/org/acme/domain/RoomResource.java rename to backend/src/main/java/org/acme/rest/RoomResource.java index 7444d84..f31b2af 100644 --- a/backend/src/main/java/org/acme/domain/RoomResource.java +++ b/backend/src/main/java/org/acme/rest/RoomResource.java @@ -1,16 +1,13 @@ -package org.acme.domain; - -import java.util.List; +package org.acme.rest; import jakarta.annotation.security.RolesAllowed; import jakarta.transaction.Transactional; -import jakarta.ws.rs.Consumes; -import jakarta.ws.rs.GET; -import jakarta.ws.rs.POST; -import jakarta.ws.rs.Path; -import jakarta.ws.rs.Produces; +import jakarta.ws.rs.*; import jakarta.ws.rs.core.MediaType; import jakarta.ws.rs.core.Response; +import org.acme.domain.Room; + +import java.util.List; @Path("/rooms") @RolesAllowed({"user"}) diff --git a/backend/src/main/java/org/acme/domain/StudentResource.java b/backend/src/main/java/org/acme/rest/StudentResource.java similarity index 76% rename from backend/src/main/java/org/acme/domain/StudentResource.java rename to backend/src/main/java/org/acme/rest/StudentResource.java index 996d015..7c75d27 100644 --- a/backend/src/main/java/org/acme/domain/StudentResource.java +++ b/backend/src/main/java/org/acme/rest/StudentResource.java @@ -1,15 +1,12 @@ -package org.acme.domain; - -import java.util.List; +package org.acme.rest; import jakarta.transaction.Transactional; -import jakarta.ws.rs.Consumes; -import jakarta.ws.rs.GET; -import jakarta.ws.rs.POST; -import jakarta.ws.rs.Path; -import jakarta.ws.rs.Produces; +import jakarta.ws.rs.*; import jakarta.ws.rs.core.MediaType; import jakarta.ws.rs.core.Response; +import org.acme.domain.Student; + +import java.util.List; @Path("/students") public class StudentResource { diff --git a/backend/src/main/java/org/acme/TimetableResource.java b/backend/src/main/java/org/acme/rest/TimetableResource.java similarity index 95% rename from backend/src/main/java/org/acme/TimetableResource.java rename to backend/src/main/java/org/acme/rest/TimetableResource.java index e6f45fa..7cbedd4 100644 --- a/backend/src/main/java/org/acme/TimetableResource.java +++ b/backend/src/main/java/org/acme/rest/TimetableResource.java @@ -1,14 +1,10 @@ -package org.acme; +package org.acme.rest; import ai.timefold.solver.core.api.solver.SolverManager; import jakarta.annotation.security.RolesAllowed; import jakarta.inject.Inject; import jakarta.transaction.Transactional; -import jakarta.ws.rs.Consumes; -import jakarta.ws.rs.GET; -import jakarta.ws.rs.POST; -import jakarta.ws.rs.Path; -import jakarta.ws.rs.Produces; +import jakarta.ws.rs.*; import jakarta.ws.rs.core.MediaType; import org.acme.domain.Room; import org.acme.domain.Student; @@ -19,9 +15,8 @@ import java.time.Duration; import java.time.LocalTime; import java.util.List; -import java.util.concurrent.ExecutionException; - import java.util.UUID; +import java.util.concurrent.ExecutionException; /** * Entry to the timetabling program. @@ -135,15 +130,15 @@ public Timetable solveExample() throws ExecutionException, InterruptedException /* * During this solving phase, new Unit objects will be created with the * allotted date and Room assignment. - * + * * Currently, the 'old' Unit objects in the 'problem' variable and the * 'new' Unit objects in the 'solution' variable are stored as different * Units in the database due to our inability to control the behaviour * of solverManager.solve - * + * * i.e. after solving, there will be 2 copies of each Unit in the - * database, where the 'old' Unit has the list of students but no - * timetable assignment, while the 'new' Unit does not have the list + * database, where the 'old' Unit has the list of students but no + * timetable assignment, while the 'new' Unit does not have the list * of students enrolled, but does have the assigned date and room */ @@ -151,7 +146,7 @@ public Timetable solveExample() throws ExecutionException, InterruptedException Timetable solution = solverManager.solve("job 1", problem).getFinalBestSolution(); - solution.persist(); + solution.persist(); // saves the solution timetable and all related entities to database return solution; diff --git a/backend/src/main/java/org/acme/domain/UnitResource.java b/backend/src/main/java/org/acme/rest/UnitResource.java similarity index 78% rename from backend/src/main/java/org/acme/domain/UnitResource.java rename to backend/src/main/java/org/acme/rest/UnitResource.java index 6405508..80b269b 100644 --- a/backend/src/main/java/org/acme/domain/UnitResource.java +++ b/backend/src/main/java/org/acme/rest/UnitResource.java @@ -1,16 +1,13 @@ -package org.acme.domain; - -import java.util.List; +package org.acme.rest; import jakarta.annotation.security.RolesAllowed; import jakarta.transaction.Transactional; -import jakarta.ws.rs.Consumes; -import jakarta.ws.rs.GET; -import jakarta.ws.rs.POST; -import jakarta.ws.rs.Path; -import jakarta.ws.rs.Produces; +import jakarta.ws.rs.*; import jakarta.ws.rs.core.MediaType; import jakarta.ws.rs.core.Response; +import org.acme.domain.Unit; + +import java.util.List; @Path("/units") @RolesAllowed({"user"}) diff --git a/backend/src/main/java/org/acme/security/jpa/Startup.java b/backend/src/main/java/org/acme/security/jpa/Startup.java index 6d497e9..53d4fa3 100644 --- a/backend/src/main/java/org/acme/security/jpa/Startup.java +++ b/backend/src/main/java/org/acme/security/jpa/Startup.java @@ -5,7 +5,6 @@ import jakarta.inject.Inject; import jakarta.inject.Singleton; import jakarta.transaction.Transactional; - import org.eclipse.microprofile.config.Config; @Singleton diff --git a/backend/src/main/java/org/acme/security/jpa/User.java b/backend/src/main/java/org/acme/security/jpa/User.java index 242b2b4..d2f384b 100644 --- a/backend/src/main/java/org/acme/security/jpa/User.java +++ b/backend/src/main/java/org/acme/security/jpa/User.java @@ -22,9 +22,10 @@ public class User extends PanacheEntity { /** * Adds a new user to the database + * * @param username the username * @param password the unencrypted password (it is encrypted with bcrypt) - * @param role the comma-separated roles + * @param role the comma-separated roles */ public static void add(String username, String password, String role) { User user = new User(); diff --git a/backend/src/test/java/org/acme/solver/TimetableConstraintProviderTest.java b/backend/src/test/java/org/acme/solver/TimetableConstraintProviderTest.java index 979e96e..4d29887 100644 --- a/backend/src/test/java/org/acme/solver/TimetableConstraintProviderTest.java +++ b/backend/src/test/java/org/acme/solver/TimetableConstraintProviderTest.java @@ -1,18 +1,16 @@ package org.acme.solver; +import ai.timefold.solver.test.api.score.stream.ConstraintVerifier; +import io.quarkus.test.junit.QuarkusTest; +import jakarta.inject.Inject; +import org.acme.domain.*; +import org.junit.jupiter.api.Test; + import java.time.DayOfWeek; import java.time.Duration; import java.time.LocalTime; import java.util.List; -import org.acme.domain.*; - -import jakarta.inject.Inject; -import ai.timefold.solver.test.api.score.stream.ConstraintVerifier; -import org.junit.jupiter.api.Test; - -import io.quarkus.test.junit.QuarkusTest; - @QuarkusTest public class TimetableConstraintProviderTest { private static final Room ROOM1 = new Room("1", "building A", "campus A", 2, true); @@ -37,7 +35,7 @@ public class TimetableConstraintProviderTest { void studentConflict() { Unit firstUnit = new Unit(1, "unit1", "Course A", DAY_OF_WEEK, START_TIME, DURATION, List.of(STUDENT1), true, ROOM1); Unit conflictingUnit = new Unit(2, "unit2", "Course A", DAY_OF_WEEK, START_TIME, DURATION, List.of(STUDENT1), false, ROOM2); - Unit nonConflictingUnit = new Unit(3, "unit3","Course B", DAY_OF_WEEK, START_TIME, DURATION, List.of(STUDENT2), true, ROOM3); + Unit nonConflictingUnit = new Unit(3, "unit3", "Course B", DAY_OF_WEEK, START_TIME, DURATION, List.of(STUDENT2), true, ROOM3); ConflictingUnit conflictingUnitPair = new ConflictingUnit(firstUnit, conflictingUnit, 1); constraintVerifier.verifyThat(TimetableConstraintProvider::studentConflict) From 13becdeae64bda0367f463ab6bb9b20bb8ec7a89 Mon Sep 17 00:00:00 2001 From: Giang Vu Date: Wed, 9 Oct 2024 00:21:25 +1100 Subject: [PATCH 3/4] style: change tab name to Timetabling for VIT --- frontend/index.html | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/frontend/index.html b/frontend/index.html index 8076bf4..c063d90 100644 --- a/frontend/index.html +++ b/frontend/index.html @@ -5,7 +5,7 @@ - Vite + React + TS + Timetabling for VIT
From 90207fa17df1911e2c757e388583e84953ea37c2 Mon Sep 17 00:00:00 2001 From: Giang Vu Date: Wed, 9 Oct 2024 00:23:07 +1100 Subject: [PATCH 4/4] style: replace Vite logo with VIT logo --- frontend/index.html | 2 +- frontend/src/assets/cropped-vit-logo.png | Bin 0 -> 18508 bytes 2 files changed, 1 insertion(+), 1 deletion(-) create mode 100644 frontend/src/assets/cropped-vit-logo.png diff --git a/frontend/index.html b/frontend/index.html index c063d90..07dd708 100644 --- a/frontend/index.html +++ b/frontend/index.html @@ -2,7 +2,7 @@ - + Timetabling for VIT diff --git a/frontend/src/assets/cropped-vit-logo.png b/frontend/src/assets/cropped-vit-logo.png new file mode 100644 index 0000000000000000000000000000000000000000..761450490756ae74a7187de8828094ed3e67810d GIT binary patch literal 18508 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 pf_v5wkfQi9s+X4Cn`Rd-`VeKg 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