Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Staging #244

Open
wants to merge 112 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
112 commits
Select commit Hold shift + click to select a range
e0d7c49
fix: tutorial popover issue
wxy1203 Sep 7, 2024
363853b
fix: home page contact us
wxy1203 Sep 14, 2024
ec49f4a
Update course schema to store GitHub repository links in an array
dexter-sim Sep 23, 2024
429a5ed
Add front end Repository page and GET API call to retrieve GitHub rep…
dexter-sim Sep 24, 2024
b8183fe
Update RepositoryInfo.tsx
dexter-sim Sep 24, 2024
34e13d4
Implement add GitHub Repo Link API and functionality
dexter-sim Sep 24, 2024
c3dc3b0
Update OverviewCard.tsx
wxy1203 Sep 25, 2024
d7ce8ea
Update Analytics.tsx
wxy1203 Sep 25, 2024
464c770
Create AllTeams.tsx
wxy1203 Sep 25, 2024
53c26fd
Update Analytics.tsx
wxy1203 Sep 25, 2024
fd1b944
Update AllTeams.tsx
wxy1203 Sep 25, 2024
617e84c
Update Analytics.tsx
wxy1203 Sep 25, 2024
83f2d3a
Update AllTeams.tsx
wxy1203 Sep 25, 2024
36b795d
Update Analytics.tsx
wxy1203 Sep 25, 2024
c07dbbf
Update AllTeams.tsx
wxy1203 Sep 25, 2024
18066f2
Update Analytics.tsx
wxy1203 Sep 25, 2024
6b46f02
Update AllTeams.tsx
wxy1203 Sep 25, 2024
72a0727
Update Overview.tsx
wxy1203 Sep 25, 2024
b21286b
Update AllTeams.tsx
wxy1203 Sep 25, 2024
406f48d
Update Overview.tsx
wxy1203 Sep 25, 2024
83d0816
Update AllTeams.tsx
wxy1203 Sep 25, 2024
c19f06a
Update Overview.tsx
wxy1203 Sep 25, 2024
8884d8b
Update AllTeams.tsx
wxy1203 Sep 25, 2024
8b73bdf
Update AllTeams.tsx
wxy1203 Sep 25, 2024
51f740b
Update AllTeams.tsx
wxy1203 Sep 25, 2024
ac174c0
fix: create course form
wxy1203 Sep 25, 2024
28b4af3
fix: create course form
wxy1203 Sep 25, 2024
9d12500
Update AllTeams.tsx
wxy1203 Sep 26, 2024
9f518b5
feat: unable to navigate before adding people
wxy1203 Sep 26, 2024
c69dab0
fix: prettier check
wxy1203 Sep 26, 2024
ad75e40
feat: need to add people first
wxy1203 Sep 26, 2024
e89b298
Update PeopleInfo.tsx
wxy1203 Sep 26, 2024
c9bd128
Update Navbar.tsx
wxy1203 Sep 26, 2024
347e09e
Update .gitignore
wxy1203 Sep 27, 2024
e0a3aab
Update package.json
wxy1203 Sep 27, 2024
0da6afb
Update package.json
wxy1203 Sep 27, 2024
22cb0df
controller, routes, service for code analysis
rjkoh Oct 4, 2024
1fe3fa7
add tests for code analysis, fix github service test
rjkoh Oct 4, 2024
9cc01e5
add basic UI and backend for TA permission
rjkoh Oct 9, 2024
7f2bec5
style fixing
rjkoh Oct 9, 2024
8d21e75
remove code analysis frontend redundancies
rjkoh Oct 9, 2024
aa86d52
frontend: group code analyses by team number and time
rjkoh Oct 9, 2024
fc0eb1d
fix: environment change
wxy1203 Oct 9, 2024
6b4305e
feat: add class overview in nav bar
wxy1203 Oct 9, 2024
82bde4c
feat: data visualization in class overview
wxy1203 Oct 9, 2024
99b8866
feat: class overview data viz (2 charts)
wxy1203 Oct 9, 2024
8f0ca2e
feat: nav bar icon
wxy1203 Oct 10, 2024
d89fe7b
fix: scroll bar for course
wxy1203 Oct 10, 2024
3040d6f
fix: eslint
wxy1203 Oct 10, 2024
24f1bae
fix: eslint
wxy1203 Oct 10, 2024
771af98
fix: eslint
wxy1203 Oct 10, 2024
a2538d3
fix: eslint
wxy1203 Oct 10, 2024
4a08301
fix: eslint
wxy1203 Oct 10, 2024
6346123
fix: remove node modules
wxy1203 Oct 10, 2024
9cc788e
Merge pull request #227 from NUS-CRISP/staging
wxy1203 Oct 10, 2024
5fbf534
fix: delete env file
wxy1203 Oct 10, 2024
51f5d30
fix: delete env files
wxy1203 Oct 10, 2024
6c67f7d
Update .gitignore
wxy1203 Oct 10, 2024
4e20898
Merge branch 'staging' into non-github-orgs
dexter-sim Oct 12, 2024
a491645
Add functionality to delete repository
dexter-sim Oct 12, 2024
a2c8956
Add functionality to edit repository links
dexter-sim Oct 13, 2024
e766b74
Merge branch 'staging' into non-github-orgs
dexter-sim Oct 13, 2024
0343d6f
add overview for code analysis frontend
rjkoh Oct 15, 2024
013375b
fix scroll overflow on codeanalysis
rjkoh Oct 15, 2024
0e42442
add timeline view
rjkoh Oct 15, 2024
9193332
add prettier
rjkoh Oct 15, 2024
beb64a6
style fixes
rjkoh Oct 15, 2024
05b9b0e
style fixes
rjkoh Oct 15, 2024
6a7ba60
style
rjkoh Oct 15, 2024
2157f83
Merge branch 'staging' into non-github-orgs
dexter-sim Oct 15, 2024
de502c9
Create fetch job for public GitHub repositories
dexter-sim Oct 15, 2024
4551959
Add unit test for getRepositories in course controller
dexter-sim Oct 17, 2024
b18f25b
Add unit tests for add, edit, and remove repository for course contro…
dexter-sim Oct 17, 2024
3f5ab7b
mkdir before clone
rjkoh Oct 18, 2024
18895b5
style
rjkoh Oct 18, 2024
beb20ad
handle private repos
rjkoh Oct 18, 2024
4eac941
style
rjkoh Oct 18, 2024
22a5d65
Merge branch 'staging' into code-analysis-backend
rjkoh Oct 18, 2024
2de0b15
follow RUN_JOB_NOW
rjkoh Oct 18, 2024
f7885c8
style
rjkoh Oct 18, 2024
0dfcd02
fix backend imports
rjkoh Oct 18, 2024
a8796c1
Add unit tests for GitHub repositories API for course service
dexter-sim Oct 18, 2024
6d2f12e
Fix formatting
dexter-sim Oct 18, 2024
11a91a0
Remove repeated attempts for code frequency stats
dexter-sim Oct 19, 2024
8008e84
feat: filter the team by teamset
wxy1203 Oct 21, 2024
54f5ae9
update
wxy1203 Oct 21, 2024
0d81b08
update
wxy1203 Oct 21, 2024
bd4ee58
Merge branch 'staging' into non-github-orgs
dexter-sim Oct 21, 2024
79be9a1
feat: class_overview and team_review
wxy1203 Oct 22, 2024
0e9159c
feat: team review - milestone and issue tracker
wxy1203 Oct 22, 2024
16a13d7
Update jira.png
wxy1203 Oct 23, 2024
3dbaba7
feat: class overview
wxy1203 Oct 23, 2024
b08a94c
feat: weekly addition and deletion
wxy1203 Oct 23, 2024
07869b7
feat: team review-individual
wxy1203 Oct 23, 2024
8e1570e
fix: eslint
wxy1203 Oct 23, 2024
0a423e1
fix: eslint
wxy1203 Oct 23, 2024
a87929c
fix: eslint
wxy1203 Oct 23, 2024
050edcf
fix: eslint
wxy1203 Oct 23, 2024
5bc43e3
feat: eslint
wxy1203 Oct 23, 2024
21e5c80
Update Navbar.tsx
wxy1203 Oct 25, 2024
ce73b4b
Update cron job for non github orgs repositories
dexter-sim Oct 26, 2024
1615ac1
Use generic Octokit for non GitHub App installed owners
dexter-sim Oct 26, 2024
c100d3d
Merge pull request #225 from rjkoh/code-analysis-backend
rjkoh Nov 11, 2024
b61db60
Merge pull request #211 from NUS-CRISP/non-github-orgs
dexter-sim Nov 18, 2024
c1f5ed5
Merge branch 'staging' into data-viz
wxy1203 Nov 18, 2024
6b61dfc
Merge pull request #220 from NUS-CRISP/data-viz
wxy1203 Nov 18, 2024
e41383a
fix: bugs
wxy1203 Nov 23, 2024
2278a82
fix: eslint
wxy1203 Nov 23, 2024
36933eb
fix: eslint
wxy1203 Nov 23, 2024
25d0d72
Merge pull request #243 from NUS-CRISP/data-viz
wxy1203 Nov 23, 2024
ea9de23
improve testing for code analysis service and controller
rjkoh Dec 18, 2024
6d146b1
Merge pull request #245 from rjkoh/code-analysis-test
rjkoh Dec 18, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
add tests for code analysis, fix github service test
  • Loading branch information
rjkoh committed Oct 4, 2024
commit 1fe3fa755f1ce24a621bb0e1de6f26d3b3706938
10 changes: 5 additions & 5 deletions backend/controllers/codeAnalysisController.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,9 @@ import {
fetchAllCodeAnalysisData,
fetchAllCodeAnalysisDataForOrg,
getAuthorizedCodeAnalysisDataByCourse,
} from 'services/codeAnalysisService';
import { MissingAuthorizationError, NotFoundError } from 'services/errors';
import { getAccountId } from 'utils/auth';
} from '../services/codeAnalysisService';
import { MissingAuthorizationError, NotFoundError } from '../services/errors';
import { getAccountId } from '../utils/auth';

export const getAllCodeAnalysisData = async (req: Request, res: Response) => {
try {
Expand All @@ -32,10 +32,10 @@ export const getAllCodeAnalysisDataByOrg = async (
if (error instanceof NotFoundError) {
res.status(404).json({ message: error.message });
} else {
console.error('Error retrieving team datas for org:', error);
console.error('Error retrieving code analysis datas for org:', error);
return res
.status(500)
.json({ error: 'Failed to get team datas for org' });
.json({ error: 'Failed to get code analysis datas for org' });
}
}
};
Expand Down
107 changes: 107 additions & 0 deletions backend/test/controllers/codeAnalysisController.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,107 @@
import { Request, Response } from 'express';
import {
getAllCodeAnalysisData,
getAllCodeAnalysisDataByOrg,
} from '../../controllers/codeAnalysisController';
import { NotFoundError } from '../../services/errors';
import * as codeAnalysisService from '../../services/codeAnalysisService';

jest.mock('../../services/codeAnalysisService');

beforeEach(() => {
jest.clearAllMocks();
});

const mockRequest = (body = {}, params = {}, query = {}) => {
const req = {} as Request;
req.body = body;
req.params = params;
req.query = query;
return req;
};

const mockResponse = () => {
const res = {} as Response;
res.status = jest.fn().mockReturnValue(res);
res.json = jest.fn().mockReturnValue(res);
return res;
};

describe('codeAnalysisController', () => {
it('should retrieve all code analysis data and send a 200 status', async () => {
const req = mockRequest();
const res = mockResponse();
const mockCodeAnalysisData = [{ teamId: 'team1', data: 'data1' }];

jest
.spyOn(codeAnalysisService, 'fetchAllCodeAnalysisData')
.mockResolvedValue(mockCodeAnalysisData as any);

await getAllCodeAnalysisData(req, res);

expect(res.status).toHaveBeenCalledWith(200);
expect(res.json).toHaveBeenCalledWith({ codeAnalysisData: mockCodeAnalysisData });
});

it('should handle errors when retrieving all code analysis data', async () => {
const req = mockRequest();
const res = mockResponse();
const error = new Error('Error fetching code analysis data');

jest.spyOn(codeAnalysisService, 'fetchAllCodeAnalysisData').mockRejectedValue(error);

await getAllCodeAnalysisData(req, res);

expect(res.status).toHaveBeenCalledWith(500);
expect(res.json).toHaveBeenCalledWith({
error: 'Failed to get all code analysis data',
});
});
});

describe('codeAnalysisController', () => {
describe('getAllCodeAnalysisDataByOrg', () => {
it('should retrieve all code analysis data for an organization and send a 200 status', async () => {
const req = mockRequest({ gitHubOrgName: 'org1' });
const res = mockResponse();
const mockCodeAnalysisData = [{ teamId: 'team1', data: 'data1' }];

jest
.spyOn(codeAnalysisService, 'fetchAllCodeAnalysisDataForOrg')
.mockResolvedValue(mockCodeAnalysisData as any);

await getAllCodeAnalysisDataByOrg(req, res);

expect(res.status).toHaveBeenCalledWith(200);
expect(res.json).toHaveBeenCalledWith(mockCodeAnalysisData);
});

it('should handle errors when retrieving all code analysis data for an organization', async () => {
const req = mockRequest({ gitHubOrgName: 'org1' });
const res = mockResponse();
const error = new NotFoundError('Code analysis data not found');

jest.spyOn(codeAnalysisService, 'fetchAllCodeAnalysisDataForOrg').mockRejectedValue(error);

await getAllCodeAnalysisDataByOrg(req, res);

expect(res.status).toHaveBeenCalledWith(404);
expect(res.json).toHaveBeenCalledWith({ message: error.message });
});

it('should handle errors when retrieving all code analysis data for an organization', async () => {
const req = mockRequest({ gitHubOrgName: 'org1' });
const res = mockResponse();
const error = new Error('Error fetching code analysis data');

jest.spyOn(codeAnalysisService, 'fetchAllCodeAnalysisDataForOrg').mockRejectedValue(error);

await getAllCodeAnalysisDataByOrg(req, res);

expect(res.status).toHaveBeenCalledWith(500);
expect(res.json).toHaveBeenCalledWith({
error: 'Failed to get code analysis datas for org',
});
});
});
});
63 changes: 63 additions & 0 deletions backend/test/models/codeAnalysisData.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
import codeAnalysisDataModel from '@models/CodeAnalysisData';
import { MongoMemoryServer } from 'mongodb-memory-server';
import mongoose, { ConnectOptions } from 'mongoose';

let mongoServer: MongoMemoryServer;

beforeAll(async () => {
mongoServer = await MongoMemoryServer.create();
const mongoUri = mongoServer.getUri();
await mongoose.connect(mongoUri, {
useNewUrlParser: true,
useUnifiedTopology: true,
} as ConnectOptions);
});

beforeEach(async () => {
await codeAnalysisDataModel.deleteMany({});
});

afterAll(async () => {
await mongoose.disconnect();
await mongoServer.stop();
});

describe('CodeAnalysisDataModel', () => {
it('should create and save a new codeAnalysisData', async () => {
const codeAnalysisData = new codeAnalysisDataModel({
executionTime: new Date(),
gitHubOrgName: 'testOrg',
teamId: 1,
repoName: 'testRepo',
metrics: ['testMetric'],
values: ['testValue'],
types: ['testType'],
domains: ['testDomain'],
});
const savedCodeAnalysisData = await codeAnalysisData.save();

expect(savedCodeAnalysisData._id).toBeDefined();
expect(savedCodeAnalysisData.executionTime).toBeInstanceOf(Date);
expect(savedCodeAnalysisData.gitHubOrgName).toBe('testOrg');
expect(savedCodeAnalysisData.teamId).toBe(1);
expect(savedCodeAnalysisData.repoName).toBe('testRepo');
expect(savedCodeAnalysisData.metrics).toEqual(['testMetric']);
expect(savedCodeAnalysisData.values).toEqual(['testValue']);
expect(savedCodeAnalysisData.types).toEqual(['testType']);
expect(savedCodeAnalysisData.domains).toEqual(['testDomain']);
});

it('should fail to save a codeAnalysisData with missing required fields', async () => {
const codeAnalysisData = new codeAnalysisDataModel({
gitHubOrgName: 'testOrg',
teamId: 1,
repoName: 'testRepo',
metrics: ['testMetric'],
values: ['testValue'],
types: ['testType'],
domains: ['testDomain'],
});

await expect(codeAnalysisData.save()).rejects.toThrow();
});
});
182 changes: 182 additions & 0 deletions backend/test/services/codeAnalysisService.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,182 @@
import AccountModel from '@models/Account';
import codeAnalysisDataModel from '@models/CodeAnalysisData';
import CourseModel from '@models/Course';
import UserModel from '@models/User';
import { MongoMemoryServer } from 'mongodb-memory-server';
import mongoose from 'mongoose';
import * as codeAnalysisService from '../../services/codeAnalysisService';
import { NotFoundError } from '../../services/errors';

let mongo: MongoMemoryServer;

beforeAll(async () => {
mongo = await MongoMemoryServer.create();
const mongoUri = mongo.getUri();
await mongoose.connect(mongoUri);
});

beforeEach(async () => {
const collections = await mongoose.connection.db.collections();
for (const collection of collections) {
await collection.deleteMany({});
}
});

afterAll(async () => {
if (mongo) {
await mongo.stop();
}
await mongoose.connection.close();
});

describe('codeAnalysisService', () => {
let mockFacultyCourseId: string;
let mockFacultyAccountId: string;

beforeEach(async () => {
await UserModel.deleteMany({});
await AccountModel.deleteMany({});
await CourseModel.deleteMany({});
await codeAnalysisDataModel.deleteMany({});

const mockFacultyUser = new UserModel({
identifier: 'test',
name: 'Test User',
enrolledCourses: [],
gitHandle: 'test',
});
await mockFacultyUser.save();
const mockFacultyAccount = new AccountModel({
email: '[email protected]',
password: 'hashedpassword',
role: 'Faculty member',
user: mockFacultyUser._id,
isApproved: true,
});
await mockFacultyAccount.save();
const mockCourse = new CourseModel({
name: 'testCourse',
code: 'testCourse',
semester: 'testCourse',
startDate: new Date(),
faculty: [mockFacultyUser._id],
TAs: [],
students: [],
teamSets: [],
courseType: 'GitHubOrg',
gitHubOrgName: 'org',
repoNameFilter: '',
installationId: 12345,
});
await mockCourse.save();

mockFacultyAccountId = mockFacultyAccount._id.toString();
mockFacultyCourseId = mockCourse._id.toString();

const codeAnalysisData1 = new codeAnalysisDataModel({
executionTime: new Date(),
gitHubOrgName: 'org',
teamId: 1,
repoName: 'team1',
metrics: ['testMetric'],
values: ['testValue'],
types: ['testType'],
domains: ['testDomain'],
});

const codeAnalysisData2 = new codeAnalysisDataModel({
executionTime: new Date(),
gitHubOrgName: 'org',
teamId: 2,
repoName: 'team2',
metrics: ['testMetric'],
values: ['testValue'],
types: ['testType'],
domains: ['testDomain'],
});

await codeAnalysisData1.save();
await codeAnalysisData2.save();
});

describe('fetchAllCodeAnalysisData', () => {
it('should fetch all code analysis data', async () => {
const result = await codeAnalysisService.fetchAllCodeAnalysisData();

expect(result).toEqual(
expect.arrayContaining([
expect.objectContaining({ repoName: 'team1' }),
expect.objectContaining({ repoName: 'team2' }),
])
);
});
});

describe('fetchAllCodeAnalysisDataForOrg', () => {
it('should throw NotFoundError if no code analysis data found for org', async () => {
await expect(
codeAnalysisService.fetchAllCodeAnalysisDataForOrg('nonexistentOrg')
).rejects.toThrow(NotFoundError);
});

it('should return code analysis datas for a given org', async () => {
const result =
await codeAnalysisService.fetchAllCodeAnalysisDataForOrg('org');

expect(result).toEqual(
expect.arrayContaining([
expect.objectContaining({ repoName: 'team1' }),
expect.objectContaining({ repoName: 'team2' })
])
);
});
});

describe('getAuthorizedCodeAnalysisDataByCourse', () => {
it('should throw NotFoundError if account not found', async () => {
await expect(
codeAnalysisService.getAuthorizedCodeAnalysisDataByCourse(
new mongoose.Types.ObjectId().toString(),
mockFacultyCourseId
)
).rejects.toThrow(NotFoundError);
});

it('should throw NotFoundError if course not found', async () => {
await expect(
codeAnalysisService.getAuthorizedCodeAnalysisDataByCourse(
mockFacultyAccountId,
new mongoose.Types.ObjectId().toString()
)
).rejects.toThrow(NotFoundError);
});

it('should throw NotFoundError if faculty member is not authorized to view code analysis data', async () => {
const unauthorizedFacultyUser = new UserModel({
identifier: 'test1',
name: 'Test User 1',
enrolledCourses: [],
gitHandle: 'test1',
});
const unauthorizedFacultyAccount = new AccountModel({
email: '[email protected]',
password: 'hashedpassword',
role: 'Faculty member',
user: unauthorizedFacultyUser._id,
isApproved: true,
});
await unauthorizedFacultyUser.save();
await unauthorizedFacultyAccount.save();

await expect(
codeAnalysisService.getAuthorizedCodeAnalysisDataByCourse(
unauthorizedFacultyAccount._id,
mockFacultyCourseId
)
).rejects.toThrow(NotFoundError);

await unauthorizedFacultyUser.deleteOne();
await unauthorizedFacultyAccount.deleteOne();
});
});
});
4 changes: 2 additions & 2 deletions backend/test/services/githubService.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -96,8 +96,8 @@ describe('gitHubService', () => {
commits: 0,
teamId: 1,
});
teamData1.save();
teamData2.save();
await teamData1.save();
await teamData2.save();
});

describe('fetchAllTeamData', () => {
Expand Down