Skip to content

Commit

Permalink
Merge branch 'dev' into submissions_student_testen
Browse files Browse the repository at this point in the history
# Conflicts:
#	frontend/src/views/GroupView.vue
  • Loading branch information
masinnae committed May 20, 2024
2 parents 0d21839 + 3280435 commit 1194de1
Show file tree
Hide file tree
Showing 14 changed files with 75 additions and 27 deletions.
44 changes: 44 additions & 0 deletions backend/alembic/versions/937c04aa37a1_add_group_num_column.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
"""add group num column
Revision ID: 937c04aa37a1
Revises: 566f33fb161f
Create Date: 2024-05-18 18:09:30.734349
"""

from typing import Sequence, Union

from alembic import op
import sqlalchemy as sa


# revision identifiers, used by Alembic.
revision: str = "937c04aa37a1"
down_revision: Union[str, None] = "566f33fb161f"
branch_labels: Union[str, Sequence[str], None] = None
depends_on: Union[str, Sequence[str], None] = None


def upgrade() -> None:
# ### commands auto generated by Alembic - please adjust! ###
op.add_column(
"team", sa.Column("num", sa.Integer(), nullable=False, server_default="0")
)
op.drop_column("team", "team_name")
# ### end Alembic commands ###


def downgrade() -> None:
# ### commands auto generated by Alembic - please adjust! ###
op.add_column(
"team",
sa.Column(
"team_name",
sa.VARCHAR(),
autoincrement=False,
nullable=False,
server_default="DEFAULT_GROUP_NAME",
),
)
op.drop_column("team", "num")
# ### end Alembic commands ###
3 changes: 2 additions & 1 deletion backend/src/group/dependencies.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,8 @@ async def retrieve_groups_by_user(
async def retrieve_groups_by_project(
project_id: int, db: AsyncSession = Depends(get_async_db)
) -> GroupList:
groups = await service.get_groups_by_project(db, project_id)
groups = list(await service.get_groups_by_project(db, project_id))
groups.sort(key=lambda x: x.num)
return GroupList(groups=groups)


Expand Down
11 changes: 9 additions & 2 deletions backend/src/group/models.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
from sqlalchemy import Column, ForeignKey, Table
from sqlalchemy import Column, ForeignKey, Table, ForeignKeyConstraint, Integer, event, select, func
from sqlalchemy.orm import Mapped, mapped_column, relationship
from src.database import Base
from typing import List
Expand All @@ -17,9 +17,16 @@ class Group(Base):
__tablename__ = "team"

id: Mapped[int] = mapped_column(primary_key=True)
team_name: Mapped[str] = mapped_column(nullable=False)
num: Mapped[int] = mapped_column(nullable=False)
score: Mapped[int] = mapped_column(nullable=False)
project_id: Mapped[int] = mapped_column(
ForeignKey("project.id", ondelete="CASCADE"), nullable=False
)
members: Mapped[List["User"]] = relationship(secondary=StudentGroup, lazy="joined")


@event.listens_for(Group, "before_insert")
def set_id(_, connect, target: Group):
query = select(func.max(Group.num)).where(Group.project_id == target.project_id)
max_id = connect.execute(query).scalar()
target.num = (max_id or 0) + 1
2 changes: 1 addition & 1 deletion backend/src/group/schemas.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,6 @@ class Groupbase(BaseModel):
model_config = ConfigDict(from_attributes=True)
project_id: int
score: int = 0
team_name: str = Field(min_length=1)


class GroupCreate(Groupbase):
Expand All @@ -17,6 +16,7 @@ class GroupCreate(Groupbase):

class Group(Groupbase):
id: int
num: int
members: list[User]


Expand Down
5 changes: 2 additions & 3 deletions backend/tests/test_group.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@
"requirements": [],
"test_files": [],
}
group_data = {"team_name": "test group", "project_id": 0}
group_data = {"project_id": 0}


@pytest_asyncio.fixture
Expand All @@ -41,14 +41,13 @@ async def group_id(client: AsyncClient, db: AsyncSession, project_id: int):

@pytest.mark.asyncio
async def test_create_group(client: AsyncClient, db: AsyncSession, project_id: int):
group_data = {"team_name": "test group", "project_id": project_id}
group_data = {"project_id": project_id}
response = await client.post("/api/groups/", json=group_data)
assert response.status_code == 403

await set_teacher(db, "test", True)
response = await client.post("/api/groups/", json=group_data)
assert response.status_code == 201 # Created
assert response.json()["team_name"] == group_data["team_name"]


@pytest.mark.asyncio
Expand Down
5 changes: 4 additions & 1 deletion frontend/src/components/groups/GroupCard.vue
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,10 @@
<v-card class="groupcard" variant="flat">
<v-row>
<v-col cols="7">
<StudentsDialog :students="group.members" :title="group.team_name" />
<StudentsDialog
:students="group.members"
:title="$t('project.group', { number: group.num })"
/>
<v-btn v-if="isTeacher" variant="flat" @click="toGroupPage">
{{ $t("group.to_grouppage") }}
</v-btn>
Expand Down
2 changes: 1 addition & 1 deletion frontend/src/components/project/ProjectSideBar.vue
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@
</router-link>
<router-link v-if="group && !isSoloProject && !isTeacher" :to="`/groups/${group!.id}`">
<v-btn class="group-button" prepend-icon="mdi-account-group">
{{ $t("project.group", { number: group!.id }) }}
{{ $t("project.group", { number: group!.num }) }}
</v-btn>
</router-link>
<router-link v-else-if="!isSoloProject && !isTeacher" :to="`/project/${project!.id}/groups`">
Expand Down
3 changes: 1 addition & 2 deletions frontend/src/models/Group.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,12 @@ import type User from "@/models/User";
export default interface Group {
id: number;
project_id: number;
num: number;
score: number;
team_name: string;
members: User[];
}

export interface GroupForm {
project_id: number;
score: number;
team_name: string;
}
4 changes: 1 addition & 3 deletions frontend/src/views/CreateProjectView.vue
Original file line number Diff line number Diff line change
Expand Up @@ -163,18 +163,16 @@ async function submitForm() {
const emptyGroup: GroupForm = {
project_id: createdProjectId,
score: 0,
team_name: "Group 1",
};
await createGroupsMutation.mutateAsync({
projectId: createdProjectId,
groups: [emptyGroup],
});
} else if (selectedGroupProject.value === "random") {
const groups = divideStudentsIntoGroups(studentsData.value || [], capacity.value);
const groupsToCreate = groups.map((_, i) => ({
const groupsToCreate = groups.map((_) => ({
project_id: createdProjectId,
score: 0,
team_name: "Group " + (i + 1),
}));
const createdGroups = await createGroupsMutation.mutateAsync({
projectId: createdProjectId,
Expand Down
8 changes: 2 additions & 6 deletions frontend/src/views/GroupView.vue
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@
</h1>
<v-row v-else>
<v-col class="col-sm-12 col-md-6 col-lg-8">
<h1>{{ group!.team_name }}</h1>
<h1>{{ $t("project.group", { number: group!.num }) }}</h1>
<h2>{{ "Project: " + project!.name }}</h2>
<v-card variant="flat">
<v-card-item :title="$t('group.members')">
Expand All @@ -24,11 +24,7 @@
color="red"
variant="flat"
@click="
() =>
removeStudent({
groupId: group!.id,
uid: member.uid,
})
() => removeStudent({ groupId: group!.id, uid: member.uid })
"
>
{{ $t("group.remove") }}</v-btn
Expand Down
7 changes: 5 additions & 2 deletions frontend/src/views/GroupsView.vue
Original file line number Diff line number Diff line change
Expand Up @@ -98,7 +98,11 @@ const isTeacher = computed(() => {
if (!user.value || !instructors.value) {
return false;
}
return instructors.value.some((instructor) => instructor.uid === user.value.uid);
return (
user.value.is_teacher ||
user.value.is_admin ||
instructors.value.some((instructor) => instructor.uid === user.value.uid)
);
});
const { mutateAsync: createGroupMutate } = useCreateGroupsMutation();
Expand All @@ -107,7 +111,6 @@ async function createGroup() {
const groupForm: GroupForm = {
project_id: project.value!.id,
score: 0,
team_name: "Group " + groups.value!.length, // Set the default team name or prompt the user for input
};
try {
Expand Down
3 changes: 1 addition & 2 deletions frontend/tests/components/buttons/GroupButtons.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,7 @@ import {ref} from "vue";
const mockGroup = {
members: [
{uid: "student1"}
],
team_name: "testgroep"
]
}

const mockMembers = [
Expand Down
3 changes: 1 addition & 2 deletions frontend/tests/components/groups/GroupCard.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,8 +17,7 @@ vi.mock("@/components/buttons/GroupButtons.vue", () => ({
const mockGroup = {
members: [
{uid: "student1"}
],
team_name: "testgroep"
]
}

const mockProject = {
Expand Down
2 changes: 1 addition & 1 deletion frontend/tests/views/GroupsView.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ const mockProject = {
}

const mockGroups = [
{id: 1, team_name: "Group 1"}
{id: 1}
]

const mockUser = {
Expand Down

0 comments on commit 1194de1

Please sign in to comment.