A Node.js library to get the schedule of the IUT Informatique of Limoges.
You can install this library via your favorite package manager.
# pnpm
pnpm add edt-iut-info-limoges
# Yarn
yarn add edt-iut-info-limoges
# npm
npm install edt-iut-info-limoges --save
TypeScript types are included in this package.
Here, we'll use year
A1
.
import { YEARS, getTimetableEntries } from "edt-iut-info-limoges";
// Get all the timetable entries for the year A1.
const timetables = await getTimetableEntries(YEARS.A1);
timetables
will be an array of TimetableEntry
objects sorted by their property week_number
.
Here, we'll use year
A1
.
import { YEARS, getLatestTimetableEntry } from "edt-iut-info-limoges";
// Get the latest timetable entry for the year A1.
const timetableEntry = await getLatestTimetableEntry(YEARS.A1);
// Parse the timetable from the timetable entry.
// Internally, will fetch the buffer of the PDF file and parse it.
const timetable = await timetableEntry.getTimetable();
Since the timetables are sorted by their week_number
property, the latest timetable will be the last entry of the getTimetableEntries()
array.
timetableEntry
will be a TimetableEntry
object and timetable
will be a Timetable
interface.
Here, we'll get a timetable entry and get its buffer to later parse it.
You can of course, provide your own buffer, from another source, here it's just to demonstrate.
import { YEARS, getLatestTimetableEntry, getTimetableFromBuffer } from "edt-iut-info-limoges";
const timetableEntry = await getLatestTimetableEntry(YEARS.A1);
// Fetch the PDF file and return the buffer.
const pdfBuffer = await timetableEntry.getBuffer();
// Parse the timetable from the buffer.
const timetable = getTimetableFromBuffer(pdfBuffer);
timetable
will be a Timetable
interface.
Here, we'll be
G1A
, so group1
and subgroupA
.
import { getLatestTimetableEntry, YEARS, SUBGROUPS, LESSON_TYPES } from "edt-iut-info-limoges";
const timetableEntry = await getLatestTimetableEntry(YEARS.A1);
const timetable = await timetableEntry.getTimetable();
const lessonsForG1A = timetable.lessons.filter(lesson => {
let isForG1A = false;
switch (lesson.type) {
case LESSON_TYPES.TP:
// We only want to keep the TP lessons that are
// for the subgroup A and the group 1.
isForG1A = lesson.group.sub === SUBGROUPS.A && lesson.group.main === 1;
break;
// Since TD lessons are the whole group, we don't
// need to check the subgroup.
case LESSON_TYPES.TD:
case LESSON_TYPES.DS:
case LESSON_TYPES.SAE:
// It's for everyone.
if (typeof lesson.group === "undefined") {
isForUser = true;
break;
}
// There's a specific subgroup can happen on SAEs.
else if (lesson.type === LESSON_TYPES.SAE && typeof lesson.group.sub !== "undefined") {
// So in that case we should check for main group and subgroup.
isForUser = lesson.group.main === 1 && lesson.group.sub === SUBGROUPS.A;
break;
}
// Otherwise, we just check for the main group.
else {
isForUser = lesson.group.main === 1;
break;
}
// Since CM and OTHER lessons are for the whole year, we don't
// need to check any group and/or subgroup.
case LESSON_TYPES.CM:
case LESSON_TYPES.OTHER:
isForG1A = true;
break;
}
return isForG1A;
});
An helper enum containing all the available years.
import { YEARS } from "edt-iut-info-limoges";
console.log(YEARS.A1); // "A1"
console.log(YEARS.A2); // "A2"
console.log(YEARS.A3); // "A3"
An helper enum containing all the available subgroups.
In the API, the subgroups are represented by a number while they are represented by a letter in real life. This enum is here to help you convert the letters to numbers.
import { SUBGROUPS } from "edt-iut-info-limoges";
console.log(SUBGROUPS.A); // 0
console.log(SUBGROUPS.B); // 1
An helper enum containing all the available lesson types.
You can use this enum to filter the lessons using their type
property.
import { LESSON_TYPES } from "edt-iut-info-limoges";
console.log(LESSON_TYPES.CM); // "CM"
console.log(LESSON_TYPES.TD); // "TD"
console.log(LESSON_TYPES.TP); // "TP"
console.log(LESSON_TYPES.DS); // "DS"
console.log(LESSON_TYPES.SAE); // "SAE"
console.log(LESSON_TYPES.OTHER); // "OTHER", used for unknown lessons
A class representing a timetable entry.
interface TimetableEntry {
// Includes the `.pdf` extension.
file_name: string;
// Date displayed on the FTP, corresponds to the last update made to the file.
// Where DateTime is a class from the `luxon` package.
last_updated: DateTime;
// From the beginning of the school year. Usually starts from September.
week_number: number;
// The year specified to get the entry, eg.: `A1`.
from_year: YEARS;
/** The direct link to the timetable. */
link: string;
// Fetch the buffer of the PDF file and returns it.
getBuffer(): Promise<Buffer>;
// Fetch the buffer of the PDF file, parse it and returns the timetable.
getTimetable(): Promise<Timetable>;
// Get the date of the last update made to the PDF file.
// Where DateTime is a class from the `luxon` package.
lastUpdated(): Promise<DateTime>;
}
Can be obtained using the getTimetableEntries()
and getLatestTimetableEntry()
functions.
An interface representing a timetable.
interface Timetable {
header: {
// Week number from September.
week_number: number;
// Week number from the beginning of the year.
week_number_in_year: number;
// DateTime is a class from the `luxon` package.
// The date of the first day of the week.
start_date: DateTime;
// The date of the last day of the week.
end_date: DateTime;
}
// The lessons of the timetable.
lessons: TimetableLesson[];
}
An interface representing a lesson.
You can use the type
property to filter the lessons using the LESSON_TYPES
enum. Each lesson type has its own properties.
interface TimetableLessonCM {
type: LESSON_TYPES.CM;
content: {
// eg.: "R1.01"
type: string;
// Name of the lesson (eg.: "Initiation au développement") from the parsed timetable.
raw_lesson: string;
// Lesson name in the official reference.
lesson_from_reference?: string;
// Name of the teacher
teacher: string;
// eg.: "AC"
room: string;
}
}
interface TimetableLessonTP {
type: LESSON_TYPES.TP;
// eg.: If you're in G1A, `main` will be `1` and the
// subgroup will be `SUBGROUPS.A` (where `SUBGROUPS.A === 0`)
group: {
main: number;
sub: SUBGROUPS;
}
content: {
// eg.: "R1.01"
type: string;
// Name of the teacher
teacher: string;
// eg.: "105"
room: string;
// Lesson name in the official reference.
lesson_from_reference?: string;
}
}
interface TimetableLessonTD {
type: LESSON_TYPES.TD;
// eg.: If you're in G1, `main` will be `1`.
group: {
main: number;
}
content: {
// eg.: "R1.01"
type: string;
// Name of the teacher
teacher: string;
// eg.: "204"
room: string;
// Lesson name in the official reference.
lesson_from_reference?: string;
}
}
interface TimetableLessonDS {
type: LESSON_TYPES.DS;
// eg.: If you're in G1, `main` will be `1`.
group: {
main: number;
}
content: {
// eg.: "R1.01"
type: string;
// Name of the teacher
teacher: string;
// eg.: "AmphiB"
room: string;
// Lesson name in the official reference.
lesson_from_reference?: string;
}
}
interface TimetableLessonSAE {
type: LESSON_TYPES.SAE;
// When `undefined`, it means that it's for every groups.
group: {
// eg.: If you're in G1, `main` will be `1`.
main: number;
/** When `undefined`, it means that it's for the whole group. */
sub?: SUBGROUPS;
} | undefined
content: {
// eg.: "S1.01"
type: string;
// Most of the time, it's the group name.
teacher: string;
// Lesson name in the official reference.
lesson_from_reference?: string;
// Lesson name parsed in the timetable, if exists.
raw_lesson?: string;
// eg.: "204"
room: string;
}
}
interface TimetableLessonOTHER {
type: LESSON_TYPES.OTHER;
content: {
description: string;
teacher: string;
room: string;
}
}
// The final type of a lesson.
type TimetableLesson = {
start_date: DateTime;
end_date: DateTime;
} & (
| TimetableLessonCM
| TimetableLessonTP
| TimetableLessonTD
| TimetableLessonDS
| TimetableLessonSAE
| TimetableLessonOTHER
);
This project wouldn't have been possible without them.
- IUT Informatique of Limoges for existing ;
pdf2json
that I forked for parsing the PDF files ;pdf.js
that is used bypdf2json
;luxon
for parsing the dates.