1
0
Fork 0
mirror of https://github.com/classchartsapi/classcharts-api-js.git synced 2026-05-14 03:56:59 +00:00
classcharts-api-js/src/core/studentClient.ts

95 lines
2.9 KiB
TypeScript
Raw Normal View History

import { API_BASE_STUDENT, BASE_URL } from "../utils/consts.ts";
import { BaseClient } from "../core/baseClient.ts";
import { parseCookies } from "../utils/utils.ts";
import { RewardPurchaseResponse, RewardsResponse } from "../types.ts";
2023-04-16 20:47:40 +01:00
2022-03-12 11:30:03 +00:00
/**
2023-04-07 13:47:08 +01:00
* Student Client
2022-03-12 11:30:03 +00:00
*/
2023-04-16 20:47:40 +01:00
export class StudentClient extends BaseClient {
/**
2023-05-14 00:59:21 +01:00
* @property studentCode ClassCharts student code
2023-04-16 20:47:40 +01:00
*/
private studentCode = "";
/**
* @property dateOfBirth Student's date of birth
*/
private dateOfBirth = "";
2022-03-12 11:30:03 +00:00
/**
2023-05-14 00:59:21 +01:00
* @param studentCode ClassCharts student code
2022-03-12 11:30:03 +00:00
* @param dateOfBirth Student's date of birth
*/
constructor(studentCode: string, dateOfBirth?: string) {
super(API_BASE_STUDENT);
2022-03-12 11:30:03 +00:00
this.studentCode = String(studentCode);
this.dateOfBirth = String(dateOfBirth);
}
/**
2023-05-14 00:59:21 +01:00
* Authenticates with ClassCharts
2022-03-12 11:30:03 +00:00
*/
async login(): Promise<void> {
2023-08-30 12:28:49 +01:00
if (!this.studentCode) throw new Error("Student Code not provided");
2022-03-12 11:30:03 +00:00
const formData = new URLSearchParams();
formData.append("_method", "POST");
formData.append("code", this.studentCode.toUpperCase());
formData.append("dob", this.dateOfBirth);
formData.append("remember_me", "1");
2023-05-14 00:59:21 +01:00
formData.append("recaptcha-token", "no-token-available");
2023-05-30 18:37:24 +01:00
const request = await fetch(BASE_URL + "/student/login", {
2022-03-12 11:30:03 +00:00
method: "POST",
body: formData,
redirect: "manual",
2022-03-12 11:30:03 +00:00
});
2023-09-10 13:02:05 +01:00
if (request.status != 302 || !request.headers.has("set-cookie")) {
2023-08-30 12:28:49 +01:00
await request.body?.cancel(); // Make deno tests happy by closing the body, unsure whether this is needed for the actual library
throw new Error(
2023-08-30 13:43:25 +01:00
"Unauthenticated: ClassCharts didn't return authentication cookies",
);
}
const cookies = String(request.headers.get("set-cookie"));
this.authCookies = cookies.split(",");
2022-08-03 16:53:46 +01:00
const sessionCookies = parseCookies(cookies);
const sessionID = JSON.parse(
2023-08-30 13:43:25 +01:00
String(sessionCookies["student_session_credentials"]),
2022-03-12 11:30:03 +00:00
);
this.sessionId = sessionID.session_id;
2023-04-07 12:43:46 +01:00
await this.getNewSessionId();
2022-03-12 11:30:03 +00:00
const user = await this.getStudentInfo();
2023-04-07 13:47:08 +01:00
this.studentId = user.data.user.id;
2022-03-12 11:30:03 +00:00
}
2023-09-18 20:52:01 +01:00
/**
2023-09-18 20:52:01 +01:00
* Gets the available items in the current student's rewards shop
* @returns Array of purchasable items
*/
async getRewards(): Promise<RewardsResponse> {
return (
await this.makeAuthedRequest(
this.API_BASE + "/rewards/" + this.studentId,
{
method: "GET",
},
)
);
}
2023-09-18 20:52:01 +01:00
/**
* Purchase a reward item from the current student's rewards shop
* @param itemId number
2023-09-18 20:52:01 +01:00
* @returns An object containing the current student's balance and item ID purchased
*/
async purchaseReward(itemId: number): Promise<RewardPurchaseResponse> {
return (
await this.makeAuthedRequest(
this.API_BASE + "/purchase/" + itemId,
{
method: "POST",
body: `pupil_id=${this.studentId}`,
},
)
);
}
2022-03-12 11:30:03 +00:00
}