import { readFile, readdir, access } from "fs/promises"; import path from "path"; import matter from "gray-matter"; import { Client } from "ldapts"; import { serialize } from "next-mdx-remote/serialize"; import { getCurrentTerm } from "@/lib/events"; const EXECS_PATH = path.join("content", "team", "execs"); const fileType = ".md"; const { year, term } = getCurrentTerm(); const execPositions: { [name: string]: string } = { president: "President", "vice-president": "Vice President", secretary: "Assistant Vice President", treasurer: "Treasurer", sysadmin: "System Administrator", }; const positionNames: string[] = [ "president", "vice-president", "secretary", "treasurer", "sysadmin", ]; export interface Metadata { name: string; role: string; image: string; } export async function getExecNames() { if (process.env.USE_LDAP?.toLowerCase() !== "true") { return (await readdir(EXECS_PATH)) .filter((name) => name.endsWith(fileType)) .map((name) => name.slice(0, -1 * fileType.length)); } const url = "ldap://ldap1.csclub.uwaterloo.ca"; const searchDN = "ou=People,dc=csclub,dc=uwaterloo,dc=ca"; const client = new Client({ url }); // position: name const execMembers: { [name: string]: string } = {}; let formattedExec: string[] = []; try { await client.bind("", ""); const { searchEntries } = await client.search(searchDN, { scope: "sub", filter: `(&(objectClass=member)(term=${(term as string).slice( 0, 1 )}${year}))`, }); // item.position might be an array if the member has more than one position searchEntries.forEach((item) => { if (typeof item.position === "string" && item.position in execPositions) { execMembers[item.position] = item.cn as string; } else if (typeof item.position === "object") { item.position.forEach((p) => { if ((p as string) in execPositions) { execMembers[p as string] = item.cn as string; } }); } }); formattedExec = positionNames.map( (position, i) => `0${i + 1}-${execMembers[position] .split(" ")[0] .toLowerCase()}-${execMembers[position].split(" ")[1].toLowerCase()}` ); formattedExec = [...formattedExec, "06-codey"]; } finally { await client.unbind(); } return formattedExec; } export async function getExec(fileName: string, convert = true) { let content, metadata; try { const raw = await readFile(path.join(EXECS_PATH, `${fileName}${fileType}`)); ({ content, data: metadata } = matter(raw)); const image = (metadata.image as string | undefined) ?? (await getMemberImagePath(metadata.name)); return { content: convert ? await serialize(content) : content, metadata: { ...metadata, image } as Metadata, }; } catch (err) { const firstName = fileName.split("-")[1][0].toUpperCase() + fileName.split("-")[1].slice(1); const lastName = fileName.split("-")[2][0].toUpperCase() + fileName.split("-")[2].slice(1); const posOrder = fileName.split("-")[0][1]; const posName = execPositions[positionNames[Number(posOrder) - 1]]; ({ content, metadata } = { content: "Coming soon!", metadata: { name: `${firstName} ${lastName}`, role: `${posName}`, }, }); const image = await getMemberImagePath(metadata.name); return { content: convert ? await serialize(content) : content, metadata: { ...metadata, image } as Metadata, }; } } async function getImage(imgPath: string) { try { await access(path.join("public", imgPath)); return imgPath; } catch { return undefined; } } export async function getMemberImagePath(name: string) { const imgPath = path.join("images", "team", name.replace(" ", "")); const placeholder = path.join( "images", "team", "team-member-placeholder.svg" ); const img = (await getImage(imgPath + ".jpg")) ?? (await getImage(imgPath + ".png")) ?? (await getImage(imgPath + ".gif")) ?? (await getImage(imgPath + ".jpeg")) ?? placeholder; return img; }