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

feat: updater-scripts #733

Draft
wants to merge 40 commits into
base: gql
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 8 commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
90145f5
base section script
adit-bala Oct 27, 2024
101301a
modularize data fetching
adit-bala Oct 28, 2024
f523109
new `Course` & `Class` Schema
adit-bala Oct 28, 2024
7099aac
`courseParser` & `classParser`
adit-bala Oct 30, 2024
7f4e9c9
Merge branch 'gql' into feat-updater-scripts
mathhulk Nov 2, 2024
537af49
don't use terms for `course` api
adit-bala Nov 3, 2024
7b9b141
add batching from @mathhulk
adit-bala Nov 3, 2024
e8a0f2b
insert results into the database
adit-bala Nov 3, 2024
3f81826
only delete relevant objects
adit-bala Nov 4, 2024
d43c524
add mongodb loader
adit-bala Nov 4, 2024
edc75d1
update optional fields for `course`
adit-bala Nov 5, 2024
a0a6122
don't include invalid data
adit-bala Nov 5, 2024
1d6f78f
simplify `course` and `class`
adit-bala Nov 7, 2024
ff86b7c
add required fields for section
adit-bala Nov 11, 2024
e7838e9
add testing script
adit-bala Nov 11, 2024
c2e46aa
init infra changes
adit-bala Nov 17, 2024
469e691
denote required fields in typescript
adit-bala Nov 17, 2024
676a797
decouple `datapuller` infra
adit-bala Nov 21, 2024
57e1931
feat: log total errors for datapuller
adit-bala Nov 21, 2024
9867932
rm dependecy
adit-bala Nov 21, 2024
a0e165c
fix: keep `datapuller` infra in `app`
adit-bala Nov 25, 2024
85d99ba
chore: migrate logic to one file
adit-bala Nov 25, 2024
489859d
chore: add encrypted env vars
adit-bala Nov 26, 2024
d8d099d
Merge branch 'gql' into feat-updater-scripts
adit-bala Nov 26, 2024
ad36e05
feat: logging + cleanup of logs
adit-bala Nov 26, 2024
c4b729a
fix: error types
adit-bala Nov 26, 2024
e4d8e4a
chore: more detailed logging
adit-bala Nov 26, 2024
93adeab
fix: proper define
adit-bala Nov 26, 2024
5a2f5ee
fix: remove `dev` check
adit-bala Nov 26, 2024
f4f30d6
fix additional {end}
maxmwang Dec 1, 2024
4eebf84
package-lock
maxmwang Dec 1, 2024
4226250
fix {{}} whitespacing
maxmwang Dec 1, 2024
16ce976
force string tag
maxmwang Dec 1, 2024
38a4716
fix helm chart
maxmwang Dec 1, 2024
0ebffb7
add datapuller to cd, specify cronjob timezone
maxmwang Dec 2, 2024
f74d8e1
specify datapuller image tags
maxmwang Dec 2, 2024
76fad25
feat: modular cronjob entrypoint script
adit-bala Dec 2, 2024
3285ff0
fix: include datapuller-prod?
adit-bala Dec 2, 2024
0eb3d4f
fix: add command to enable `datapuller-prod`
adit-bala Dec 4, 2024
0f026b9
fix: remove unused `env`
adit-bala Dec 4, 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
2 changes: 2 additions & 0 deletions apps/datapuller/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,8 @@
"private": true,
"scripts": {
"build": "tsc --noEmit",
"section": "tsx src/section.ts",
"class": "tsx src/class.ts",
"course": "tsx src/course.ts"
},
"devDependencies": {
Expand Down
67 changes: 67 additions & 0 deletions apps/datapuller/src/class.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
import { ClassModel, IClassItem } from "@repo/common";
import { ClassesAPI } from "@repo/sis-api/classes";

import setup from "./shared";
import mapClassToNewClass, { CombinedClass } from "./shared/classParser";
import { Config } from "./shared/config";
import { fetchActiveTerms, fetchPaginatedData } from "./shared/utils";

async function updateClasses(config: Config) {
const log = config.log;
const classesAPI = new ClassesAPI();

log.info("Fetching Active Terms");
const activeTerms = await fetchActiveTerms(log, {
app_id: config.sis.TERM_APP_ID,
app_key: config.sis.TERM_APP_KEY,
});

log.info(activeTerms);

const classes = await fetchPaginatedData<IClassItem, CombinedClass>(
log,
classesAPI.v1,
activeTerms,
"getClassesUsingGet",
{
app_id: config.sis.CLASS_APP_ID,
app_key: config.sis.CLASS_APP_KEY,
},
(data) => data.apiResponse.response.classes || [],
mapClassToNewClass
);

log.info("Example Class:", classes[0]);

await ClassModel.deleteMany({});
adit-bala marked this conversation as resolved.
Show resolved Hide resolved

// Insert classes in batches of 5000
const insertBatchSize = 5000;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just leaving this here, but we should probably do some testing on hozer to determine the final batch size. It all depends on amount of memory, etc.


for (let i = 0; i < classes.length; i += insertBatchSize) {
const batch = classes.slice(i, i + insertBatchSize);

console.log(`Inserting batch ${i / insertBatchSize + 1}...`);

await ClassModel.insertMany(batch, { ordered: false });
}

console.log(`Completed updating database with new class data.`);

log.info(`Updated ${classes.length} classes for active terms`);
}

const initialize = async () => {
const { config } = setup();
try {
config.log.info("\n=== UPDATE CLASSES ===");
await updateClasses(config);
} catch (error) {
config.log.error(error);
process.exit(1);
}

process.exit(0);
};

initialize();
72 changes: 44 additions & 28 deletions apps/datapuller/src/course.ts
Original file line number Diff line number Diff line change
@@ -1,43 +1,59 @@
import { TermModel } from "@repo/common";
import { ClassesAPI } from "@repo/sis-api/classes";
import { CourseModel, ICourseItem } from "@repo/common";
import { CoursesAPI } from "@repo/sis-api/courses";
import { TermsAPI } from "@repo/sis-api/terms";

import setup from "./shared";
import { Config } from "./shared/config";
import mapCourseToNewCourse, { CombinedCourse } from "./shared/courseParser";
import { fetchPaginatedData } from "./shared/utils";

async function main() {
const { log } = setup();

// Terms API example
const termsAPI = new TermsAPI();
async function updateCourses(config: Config) {
const log = config.log;
const coursesAPI = new CoursesAPI();

await termsAPI.v2.getByTermsUsingGet(
const courses = await fetchPaginatedData<ICourseItem, CombinedCourse>(
log,
coursesAPI.v4,
null,
"findCourseCollectionUsingGet",
{
"temporal-position": "Current",
app_id: config.sis.COURSE_APP_ID,
app_key: config.sis.COURSE_APP_KEY,
},
{
headers: {
app_id: "123",
app_key: "abc",
},
}
(data) => data.apiResponse.response.courses || [],
mapCourseToNewCourse
);

// Courses API example
const coursesAPI = new CoursesAPI();
log.info("Example Course:", courses[0]);

await CourseModel.deleteMany({});

await coursesAPI.v4.findCourseCollectionUsingGet({
"last-updated-since": "2021-01-01",
});
// Insert courses in batches of 5000
const insertBatchSize = 5000;

// Classes API example
const classesAPI = new ClassesAPI();
for (let i = 0; i < courses.length; i += insertBatchSize) {
const batch = courses.slice(i, i + insertBatchSize);

await classesAPI.v1.getClassesUsingGet({
"term-id": "123",
});
console.log(`Inserting batch ${i / insertBatchSize + 1}...`);

log.info(TermModel);
await CourseModel.insertMany(batch, { ordered: false });
}

console.log(`Completed updating database with new course data.`);

log.info(`Updated ${courses.length} courses for active terms`);
}

main();
const initialize = async () => {
const { config } = setup();
try {
config.log.info("\n=== UPDATE COURSES ===");
await updateCourses(config);
} catch (error) {
config.log.error(error);
process.exit(1);
}

process.exit(0);
};

initialize();
67 changes: 67 additions & 0 deletions apps/datapuller/src/section.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
import { ISectionItem, SectionModel } from "@repo/common";
import { ClassSection, ClassesAPI } from "@repo/sis-api/classes";

import setup from "./shared";
import { Config } from "./shared/config";
import mapSectionToNewSection from "./shared/sectionParser";
import { fetchActiveTerms, fetchPaginatedData } from "./shared/utils";

async function updateSections(config: Config) {
const log = config.log;
const classesAPI = new ClassesAPI();

log.info("Fetching Active Terms");
const activeTerms = await fetchActiveTerms(log, {
app_id: config.sis.TERM_APP_ID,
app_key: config.sis.TERM_APP_KEY,
});

log.info(activeTerms);

const sections = await fetchPaginatedData<ISectionItem, ClassSection>(
log,
classesAPI.v1,
activeTerms,
"getClassSectionsUsingGet",
{
app_id: config.sis.CLASS_APP_ID,
app_key: config.sis.CLASS_APP_KEY,
},
(data) => data.apiResponse.response.classSections || [],
mapSectionToNewSection
);

log.info("Example Section:", sections[0]);

log.info(`Updated ${sections.length} sections for Spring 2024`);

await SectionModel.deleteMany({});
adit-bala marked this conversation as resolved.
Show resolved Hide resolved

// Insert sections in batches of 5000
const insertBatchSize = 5000;

for (let i = 0; i < sections.length; i += insertBatchSize) {
const batch = sections.slice(i, i + insertBatchSize);

console.log(`Inserting batch ${i / insertBatchSize + 1}...`);

await SectionModel.insertMany(batch, { ordered: false });
}

console.log(`Completed updating database with new section data.`);
}

const initialize = async () => {
const { config } = setup();
try {
config.log.info("\n=== UPDATE SECTIONS ===");
await updateSections(config);
} catch (error) {
config.log.error(error);
process.exit(1);
}

process.exit(0);
};

initialize();
106 changes: 106 additions & 0 deletions apps/datapuller/src/shared/classParser.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,106 @@
import { IClassItem } from "@repo/common";
import { Class } from "@repo/sis-api/classes";

import { getRequiredField } from "./utils";

// Include other relevant fields missing in the auto-generated Class type
export type CombinedClass = Class & {
requisites: {
code: {
type: string;
};
description: {
type: string;
};
formalDescription: string;
active: boolean;
fromDate: Date;
toDate: Date;
};
};

export default function mapClassToNewClass(
original: CombinedClass
): IClassItem {
const courseId = getRequiredField(
original.course?.identifiers?.find((i) => i.type == "cs-course-id")?.id,
"courseId",
""
);

const newClass: IClassItem = {
adit-bala marked this conversation as resolved.
Show resolved Hide resolved
courseId,
termId: getRequiredField(original.session?.term?.id, "session.term.id", ""),
sessionId: getRequiredField(original.session?.id, "session.id", ""),
number: getRequiredField(original.number, "number", ""),
offeringNumber: getRequiredField(
original.offeringNumber,
"offeringNumber",
0
),
title: getRequiredField(original.classTitle, "classTitle", ""),
description: getRequiredField(
original.classDescription,
"classDescription",
""
),
allowedUnits: {
minimum: getRequiredField(
original.allowedUnits?.minimum,
"allowedUnits.minimum",
0
),
maximum: getRequiredField(
original.allowedUnits?.maximum,
"allowedUnits.maximum",
0
),
forAcademicProgress: getRequiredField(
original.allowedUnits?.forAcademicProgress,
"allowedUnits.forAcademicProgress",
0
),
forFinancialAid: getRequiredField(
original.allowedUnits?.forFinancialAid,
"allowedUnits.forFinancialAid",
0
),
},
gradingBasis: getRequiredField(
original.gradingBasis?.code,
"gradingBasis.code",
""
),
status: getRequiredField(original.status?.code, "status.code", ""),
finalExam: getRequiredField(original.finalExam?.code, "finalExam.code", ""),
instructionMode: getRequiredField(
original.instructionMode?.code,
"instructionMode.code",
""
),
anyPrintInScheduleOfClasses: getRequiredField(
original.anyPrintInScheduleOfClasses,
"anyPrintInScheduleOfClasses",
false
),
contactHours: getRequiredField(original.contactHours, "contactHours", 0),
blindGrading: getRequiredField(
original.blindGrading,
"blindGrading",
false
),
requirementDesignation: getRequiredField(
original.requirementDesignation?.code,
"requirementDesignation.code",
""
),
// TODO: fix with proper requisites
requisites: getRequiredField(
original.requisites?.formalDescription,
"requisites.formalDescription",
""
),
};

return newClass;
}
12 changes: 12 additions & 0 deletions apps/datapuller/src/shared/config.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import dotenv from "dotenv";
import { Logger } from "tslog";

// Safely get the environment variable in the process
const env = (name: string): string => {
Expand All @@ -12,6 +13,7 @@ const env = (name: string): string => {
};

export interface Config {
log: Logger<unknown>;
isDev: boolean;
mongoDB: {
uri: string;
Expand All @@ -21,13 +23,21 @@ export interface Config {
CLASS_APP_KEY: string;
COURSE_APP_ID: string;
COURSE_APP_KEY: string;
TERM_APP_ID: string;
TERM_APP_KEY: string;
};
}

export function loadConfig(): Config {
dotenv.config();

const log = new Logger({
type: "pretty",
prettyLogTimeZone: "local",
});

return {
log,
isDev: env("NODE_ENV") === "development",
mongoDB: {
uri: env("MONGODB_URI"),
Expand All @@ -37,6 +47,8 @@ export function loadConfig(): Config {
CLASS_APP_KEY: env("SIS_CLASS_APP_KEY"),
COURSE_APP_ID: env("SIS_COURSE_APP_ID"),
COURSE_APP_KEY: env("SIS_COURSE_APP_KEY"),
TERM_APP_ID: env("SIS_TERM_APP_ID"),
TERM_APP_KEY: env("SIS_TERM_APP_KEY"),
},
};
}
Loading