From 0aa22ae31cee43078892933bea36f5721d39eead Mon Sep 17 00:00:00 2001 From: garrettladley Date: Fri, 26 Apr 2024 17:37:26 -0400 Subject: [PATCH] feat: fe lib ci --- .github/workflows/frontend_lib.yml | 76 +++++++++++++++++++++++ .github/workflows/frontend_lib_codeql.yml | 41 ++++++++++++ frontend/dashboard/.eslintrc.json | 1 - frontend/lib/.eslintrc.json | 4 ++ frontend/lib/api/category.ts | 15 ++--- frontend/lib/api/club.ts | 17 +++-- frontend/lib/api/contact.ts | 10 +-- frontend/lib/api/event.ts | 12 ++-- frontend/lib/api/index.ts | 12 ++-- frontend/lib/api/tag.ts | 50 +++++++-------- frontend/lib/api/user.ts | 42 +++++++------ frontend/lib/index.ts | 8 +-- frontend/lib/package.json | 11 +++- frontend/lib/tsconfig.json | 21 ++----- frontend/lib/types/auth.ts | 4 +- frontend/lib/types/club.ts | 20 +++--- frontend/lib/types/error.ts | 2 +- frontend/lib/types/event.ts | 24 +++---- frontend/lib/types/index.ts | 22 +++---- frontend/lib/types/item.ts | 4 +- frontend/lib/types/root.ts | 8 +-- frontend/lib/types/tag.ts | 2 +- frontend/lib/types/user.ts | 36 +++++------ 23 files changed, 281 insertions(+), 161 deletions(-) create mode 100644 .github/workflows/frontend_lib.yml create mode 100644 .github/workflows/frontend_lib_codeql.yml create mode 100644 frontend/lib/.eslintrc.json diff --git a/.github/workflows/frontend_lib.yml b/.github/workflows/frontend_lib.yml new file mode 100644 index 000000000..6b4dd6d28 --- /dev/null +++ b/.github/workflows/frontend_lib.yml @@ -0,0 +1,76 @@ +name: Frontend Lib + +permissions: read-all + +on: + push: + paths: + - frontend/lib/** + - .github/workflows/frontend_lib.yml + pull_request: + types: opened + paths: + - frontend/lib/** + - .github/workflows/frontend_lib.yml + +jobs: + format: + name: Format + runs-on: ubuntu-latest + steps: + - name: Checkout code + uses: actions/checkout@v3 + - name: Setup Node.js + uses: actions/setup-node@v3 + with: + node-version: 18.x + cache: yarn + cache-dependency-path: frontend/lib/yarn.lock + - name: Install dependencies + run: | + cd frontend/lib + yarn install + - name: Format + run: | + cd frontend/lib + yarn format + lint: + name: Lint + runs-on: ubuntu-latest + steps: + - name: Checkout code + uses: actions/checkout@v3 + - name: Setup Node.js + uses: actions/setup-node@v3 + with: + node-version: 18.x + cache: yarn + cache-dependency-path: frontend/lib/yarn.lock + - name: Install dependencies + run: | + cd frontend/lib + yarn install + - name: Lint + run: | + cd frontend/lib + yarn lint + test: + name: Test + runs-on: ubuntu-latest + steps: + - name: Checkout code + uses: actions/checkout@v3 + - name: Setup Node.js + uses: actions/setup-node@v3 + with: + node-version: 18.x + cache: yarn + cache-dependency-path: frontend/lib/yarn.lock + - name: Install dependencies + run: | + cd frontend/lib + yarn install + - name: Test + run: | + cd frontend/lib + yarn test \ No newline at end of file diff --git a/.github/workflows/frontend_lib_codeql.yml b/.github/workflows/frontend_lib_codeql.yml new file mode 100644 index 000000000..1f2fc2395 --- /dev/null +++ b/.github/workflows/frontend_lib_codeql.yml @@ -0,0 +1,41 @@ +name: Frontend Lib CodeQL + +on: + push: + paths: + - "frontend/lib/**" + - ".github/workflows/frontend_lib_codeql.yml" + pull_request: + types: [opened] + paths: + - "frontend/lib/**" + - ".github/workflows/frontend_lib_codeql.yml" + +concurrency: + group: ${{ github.workflow }}-${{ github.ref }} + cancel-in-progress: true + +jobs: + analyze: + name: Analyze + runs-on: ubuntu-latest + permissions: + security-events: write + strategy: + fail-fast: false + steps: + - name: Checkout repository + uses: actions/checkout@v4 + - name: Initialize CodeQL + uses: github/codeql-action/init@v3 + with: + languages: javascript-typescript + queries: security-and-quality + - name: Autobuild + uses: github/codeql-action/autobuild@v3 + with: + working-directory: frontend/lib + - name: Perform CodeQL Analysis + uses: github/codeql-action/analyze@v3 + with: + category: lib diff --git a/frontend/dashboard/.eslintrc.json b/frontend/dashboard/.eslintrc.json index d229e86f2..83df9d1d0 100644 --- a/frontend/dashboard/.eslintrc.json +++ b/frontend/dashboard/.eslintrc.json @@ -1,4 +1,3 @@ { - "extends": "next/core-web-vitals", "plugins": ["prettier"] } diff --git a/frontend/lib/.eslintrc.json b/frontend/lib/.eslintrc.json new file mode 100644 index 000000000..d229e86f2 --- /dev/null +++ b/frontend/lib/.eslintrc.json @@ -0,0 +1,4 @@ +{ + "extends": "next/core-web-vitals", + "plugins": ["prettier"] +} diff --git a/frontend/lib/api/category.ts b/frontend/lib/api/category.ts index 7057b77f9..86bfa592b 100644 --- a/frontend/lib/api/category.ts +++ b/frontend/lib/api/category.ts @@ -1,9 +1,9 @@ -import { z } from 'zod'; +import { z } from "zod"; -import { API_BASE_URL } from '@/const'; -import { Category, categorySchema } from '@/types/category'; -import { uuid } from '@/types/uuid'; -import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'; +import { API_BASE_URL } from "@/const"; +import { Category, categorySchema } from "@/types/category"; +import { uuid } from "@/types/uuid"; +import { createApi, fetchBaseQuery } from "@reduxjs/toolkit/query/react"; export const categoriesApi = createApi({ reducerPath: "categoriesApi", @@ -28,7 +28,4 @@ export const categoriesApi = createApi({ }), }); -export const { - useCategoriesQuery, - useCategoryQuery, -} = categoriesApi; \ No newline at end of file +export const { useCategoriesQuery, useCategoryQuery } = categoriesApi; diff --git a/frontend/lib/api/club.ts b/frontend/lib/api/club.ts index 3e2ace508..f6368028d 100644 --- a/frontend/lib/api/club.ts +++ b/frontend/lib/api/club.ts @@ -1,9 +1,9 @@ -import { API_BASE_URL } from '@/const'; -import { Club } from '@/types/club'; -import { Contact } from '@/types/contact'; -import { Tag } from '@/types/tag'; -import { uuid } from '@/types/uuid'; -import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'; +import { API_BASE_URL } from "@/const"; +import { Club } from "@/types/club"; +import { Contact } from "@/types/contact"; +import { Tag } from "@/types/tag"; +import { uuid } from "@/types/uuid"; +import { createApi, fetchBaseQuery } from "@reduxjs/toolkit/query/react"; export const clubsApi = createApi({ reducerPath: "clubsApi", @@ -41,7 +41,6 @@ export const clubsApi = createApi({ providesTags: (result, _, clubID) => result ? [{ type: "Clubs", id: clubID }] : [], }), - }), }); @@ -49,5 +48,5 @@ export const { useClubsQuery, useClubQuery, useClubContactsQuery, - useClubTagsQuery -} = clubsApi; \ No newline at end of file + useClubTagsQuery, +} = clubsApi; diff --git a/frontend/lib/api/contact.ts b/frontend/lib/api/contact.ts index a2f95b8d2..9db031cd7 100644 --- a/frontend/lib/api/contact.ts +++ b/frontend/lib/api/contact.ts @@ -1,9 +1,9 @@ -import { z } from 'zod'; +import { z } from "zod"; -import { API_BASE_URL } from '@/const'; -import { Contact, contactSchema } from '@/types/contact'; -import { uuid } from '@/types/uuid'; -import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'; +import { API_BASE_URL } from "@/const"; +import { Contact, contactSchema } from "@/types/contact"; +import { uuid } from "@/types/uuid"; +import { createApi, fetchBaseQuery } from "@reduxjs/toolkit/query/react"; export const contactsApi = createApi({ reducerPath: "contactApi", diff --git a/frontend/lib/api/event.ts b/frontend/lib/api/event.ts index 547ba1264..7105cce15 100644 --- a/frontend/lib/api/event.ts +++ b/frontend/lib/api/event.ts @@ -1,8 +1,8 @@ -import { API_BASE_URL } from '@/const'; -import { Club } from '@/types/club'; -import { Event } from '@/types/event'; -import { uuid } from '@/types/uuid'; -import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'; +import { API_BASE_URL } from "@/const"; +import { Club } from "@/types/club"; +import { Event } from "@/types/event"; +import { uuid } from "@/types/uuid"; +import { createApi, fetchBaseQuery } from "@reduxjs/toolkit/query/react"; export const eventsApi = createApi({ reducerPath: "eventsApi", @@ -48,4 +48,4 @@ export const { useEventQuery, useEventTagsQuery, useEventHostsQuery, -} = eventsApi; \ No newline at end of file +} = eventsApi; diff --git a/frontend/lib/api/index.ts b/frontend/lib/api/index.ts index 5131285ec..e6b1183be 100644 --- a/frontend/lib/api/index.ts +++ b/frontend/lib/api/index.ts @@ -1,6 +1,6 @@ -export * from './tag' -export * from './user' -export * from './event' -export * from './contact' -export * from './club' -export * from './category' \ No newline at end of file +export * from "./tag"; +export * from "./user"; +export * from "./event"; +export * from "./contact"; +export * from "./club"; +export * from "./category"; diff --git a/frontend/lib/api/tag.ts b/frontend/lib/api/tag.ts index 7a23066cb..e6da1c3b5 100644 --- a/frontend/lib/api/tag.ts +++ b/frontend/lib/api/tag.ts @@ -1,31 +1,31 @@ -import { z } from 'zod'; +import { z } from "zod"; -import { API_BASE_URL } from '@/const'; -import { Tag, tagSchema } from '@/types/tag'; -import { uuid } from '@/types/uuid'; -import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'; +import { API_BASE_URL } from "@/const"; +import { Tag, tagSchema } from "@/types/tag"; +import { uuid } from "@/types/uuid"; +import { createApi, fetchBaseQuery } from "@reduxjs/toolkit/query/react"; export const tagsApi = createApi({ - reducerPath: "tagsApi", - baseQuery: fetchBaseQuery({ baseUrl: API_BASE_URL }), - tagTypes: ["Tags"], - endpoints: (builder) => ({ - tags: builder.query({ - query: () => "tags", - transformResponse: (rawResponse: unknown) => { - return z.array(tagSchema).parse(rawResponse); - }, - providesTags: [{ type: "Tags", id: "LIST" }], - }), - tag: builder.query({ - query: (tagID: uuid) => `tags/${tagID}`, - transformResponse: (rawResponse: unknown) => { - return tagSchema.parse(rawResponse); - }, - providesTags: (result, _, tagID) => - result ? [{ type: "Tags", id: tagID }] : [], - }), + reducerPath: "tagsApi", + baseQuery: fetchBaseQuery({ baseUrl: API_BASE_URL }), + tagTypes: ["Tags"], + endpoints: (builder) => ({ + tags: builder.query({ + query: () => "tags", + transformResponse: (rawResponse: unknown) => { + return z.array(tagSchema).parse(rawResponse); + }, + providesTags: [{ type: "Tags", id: "LIST" }], }), + tag: builder.query({ + query: (tagID: uuid) => `tags/${tagID}`, + transformResponse: (rawResponse: unknown) => { + return tagSchema.parse(rawResponse); + }, + providesTags: (result, _, tagID) => + result ? [{ type: "Tags", id: tagID }] : [], + }), + }), }); -export const { useTagsQuery, useTagQuery } = tagsApi; \ No newline at end of file +export const { useTagsQuery, useTagQuery } = tagsApi; diff --git a/frontend/lib/api/user.ts b/frontend/lib/api/user.ts index b9d3a565c..bf7ead91e 100644 --- a/frontend/lib/api/user.ts +++ b/frontend/lib/api/user.ts @@ -1,22 +1,22 @@ -import { z } from 'zod'; +import { z } from "zod"; -import { API_BASE_URL } from '@/const'; -import { Club, clubSchema } from '@/types/club'; -import { User, userSchema } from '@/types/user'; -import { uuid } from '@/types/uuid'; -import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'; +import { API_BASE_URL } from "@/const"; +import { Club, clubSchema } from "@/types/club"; +import { User, userSchema } from "@/types/user"; +import { uuid } from "@/types/uuid"; +import { createApi, fetchBaseQuery } from "@reduxjs/toolkit/query/react"; export const usersApi = createApi({ reducerPath: "usersApi", baseQuery: fetchBaseQuery({ baseUrl: API_BASE_URL }), - tagTypes: ['Users'], + tagTypes: ["Users"], endpoints: (builder) => ({ users: builder.query({ query: () => "users", transformResponse: (rawResponse: unknown) => { return z.array(userSchema).parse(rawResponse); }, - providesTags: [{ type: 'Users', id: 'LIST' }], + providesTags: [{ type: "Users", id: "LIST" }], }), user: builder.query({ query: (userID: uuid) => `users/${userID}`, @@ -24,7 +24,7 @@ export const usersApi = createApi({ return userSchema.parse(rawResponse); }, providesTags: (result, _, userID) => - result ? [{ type: 'Users', id: userID }] : [], + result ? [{ type: "Users", id: userID }] : [], }), userFollowing: builder.query({ query: (userID: uuid) => `users/${userID}/follower`, @@ -32,21 +32,27 @@ export const usersApi = createApi({ return z.array(clubSchema).parse(rawResponse); }, providesTags: (result, _, userID) => - result ? [{ type: 'Users', id: userID }] : [], + result ? [{ type: "Users", id: userID }] : [], }), - createUserClubFollowing: builder.mutation({ + createUserClubFollowing: builder.mutation< + void, + { userID: uuid; clubID: uuid } + >({ query: ({ userID, clubID }) => ({ url: `users/${userID}/follower/${clubID}`, - method: 'POST', + method: "POST", }), - invalidatesTags: ['Users'], + invalidatesTags: ["Users"], }), - deleteUserClubFollowing: builder.mutation({ + deleteUserClubFollowing: builder.mutation< + void, + { userID: uuid; clubID: uuid } + >({ query: ({ userID, clubID }) => ({ url: `users/${userID}/follower/${clubID}`, - method: 'DELETE', + method: "DELETE", }), - invalidatesTags: ['Users'], + invalidatesTags: ["Users"], }), }), }); @@ -56,5 +62,5 @@ export const { useUserQuery, useUserFollowingQuery, useCreateUserClubFollowingMutation, - useDeleteUserClubFollowingMutation -} = usersApi; \ No newline at end of file + useDeleteUserClubFollowingMutation, +} = usersApi; diff --git a/frontend/lib/index.ts b/frontend/lib/index.ts index 7e663cb35..c534e8d9d 100644 --- a/frontend/lib/index.ts +++ b/frontend/lib/index.ts @@ -1,5 +1,5 @@ -export * from './api'; -export * from './types'; +export * from "./api"; +export * from "./types"; -export * from './store'; -export * from './const' \ No newline at end of file +export * from "./store"; +export * from "./const"; diff --git a/frontend/lib/package.json b/frontend/lib/package.json index 637b8db3f..d11cf9dfd 100644 --- a/frontend/lib/package.json +++ b/frontend/lib/package.json @@ -2,9 +2,18 @@ "name": "@sac/lib", "main": "index.ts", "version": "0.1.0", + "scripts": { + "test": "echo \"Woah there, we have no frontend tests as of right now. Let's just say we're passing.\" && exit 0", + "lint": "eslint . --ext .js,.jsx,.ts,.tsx", + "format": "prettier --write ." + }, "private": true, "dependencies": { "@reduxjs/toolkit": "^2.2.3", "zod": "^3.22.4" + }, + "devDependencies": { + "eslint-plugin-prettier": "^5.1.3", + "prettier": "^3.2.4" } -} \ No newline at end of file +} diff --git a/frontend/lib/tsconfig.json b/frontend/lib/tsconfig.json index 7a814810d..fd9cd89d0 100644 --- a/frontend/lib/tsconfig.json +++ b/frontend/lib/tsconfig.json @@ -2,10 +2,7 @@ "compilerOptions": { "allowSyntheticDefaultImports": true, "jsx": "react", - "lib": [ - "dom", - "esnext" - ], + "lib": ["dom", "esnext"], "experimentalDecorators": true, "emitDecoratorMetadata": true, "moduleResolution": "node", @@ -17,17 +14,9 @@ "strict": true, "baseUrl": "./", "paths": { - "@/*": [ - "./*" - ] + "@/*": ["./*"] } }, - "exclude": [ - "node_modules" - ], - "include": [ - "**/*.ts", - "**/*.tsx", - "types/index.ts" - ] -} \ No newline at end of file + "exclude": ["node_modules"], + "include": ["**/*.ts", "**/*.tsx", "types/index.ts"] +} diff --git a/frontend/lib/types/auth.ts b/frontend/lib/types/auth.ts index 30635032e..bef872336 100644 --- a/frontend/lib/types/auth.ts +++ b/frontend/lib/types/auth.ts @@ -1,4 +1,4 @@ export type Tokens = { - accessToken: string; - refreshToken: string; + accessToken: string; + refreshToken: string; }; diff --git a/frontend/lib/types/club.ts b/frontend/lib/types/club.ts index a6375f4ed..26dc2641b 100644 --- a/frontend/lib/types/club.ts +++ b/frontend/lib/types/club.ts @@ -1,16 +1,16 @@ import { z } from "zod"; -import { rootModelSchema } from '@/types/root'; +import { rootModelSchema } from "@/types/root"; const clubSchemaIntermediate = z.object({ - name: z.string().max(255), - preview: z.string().max(255), - description: z.string().max(255), - num_members: z.number(), - is_recruiting: z.boolean(), - recruitment_cycle: z.enum(["fall", "spring", "fallSpring", "always"]), - recruitment_type: z.enum(["unrestricted", "tryout", "application"]), - application_link: z.string().max(255), - logo: z.string().max(255).optional(), + name: z.string().max(255), + preview: z.string().max(255), + description: z.string().max(255), + num_members: z.number(), + is_recruiting: z.boolean(), + recruitment_cycle: z.enum(["fall", "spring", "fallSpring", "always"]), + recruitment_type: z.enum(["unrestricted", "tryout", "application"]), + application_link: z.string().max(255), + logo: z.string().max(255).optional(), }); export const clubSchema = clubSchemaIntermediate.merge(rootModelSchema); diff --git a/frontend/lib/types/error.ts b/frontend/lib/types/error.ts index aa50e46ac..6bfb33c7d 100644 --- a/frontend/lib/types/error.ts +++ b/frontend/lib/types/error.ts @@ -1,3 +1,3 @@ export type ErrorResponse = { - error: string; + error: string; }; diff --git a/frontend/lib/types/event.ts b/frontend/lib/types/event.ts index 5a0e5076b..36efdb144 100644 --- a/frontend/lib/types/event.ts +++ b/frontend/lib/types/event.ts @@ -1,18 +1,18 @@ -import { z } from 'zod'; +import { z } from "zod"; -import { rootModelSchema } from './root'; +import { rootModelSchema } from "./root"; const eventSchemaIntermediate = z.object({ - name: z.string().max(255), - preview: z.string().max(255), - content: z.string().max(255), - start_time: z.date(), - end_time: z.date(), - location: z.string().max(255), - meeting_link: z.string().max(255).optional(), - event_type: z.enum(['open', 'membersOnly']), - is_recurring: z.boolean(), - host: z.string().max(255) + name: z.string().max(255), + preview: z.string().max(255), + content: z.string().max(255), + start_time: z.date(), + end_time: z.date(), + location: z.string().max(255), + meeting_link: z.string().max(255).optional(), + event_type: z.enum(["open", "membersOnly"]), + is_recurring: z.boolean(), + host: z.string().max(255), }); export const eventSchema = eventSchemaIntermediate.merge(rootModelSchema); diff --git a/frontend/lib/types/index.ts b/frontend/lib/types/index.ts index 54c08756b..2091fd7b6 100644 --- a/frontend/lib/types/index.ts +++ b/frontend/lib/types/index.ts @@ -1,12 +1,12 @@ -export * from './auth' -export * from './item' -export * from './uuid' -export * from './error' +export * from "./auth"; +export * from "./item"; +export * from "./uuid"; +export * from "./error"; -export * from './root' -export * from './club' -export * from './contact' -export * from './user' -export * from './tag' -export * from './category' -export * from './event' \ No newline at end of file +export * from "./root"; +export * from "./club"; +export * from "./contact"; +export * from "./user"; +export * from "./tag"; +export * from "./category"; +export * from "./event"; diff --git a/frontend/lib/types/item.ts b/frontend/lib/types/item.ts index 763910602..8cef04537 100644 --- a/frontend/lib/types/item.ts +++ b/frontend/lib/types/item.ts @@ -1,4 +1,4 @@ export type Item = { - label: string; - value: string; + label: string; + value: string; }; diff --git a/frontend/lib/types/root.ts b/frontend/lib/types/root.ts index 4003c9d2b..9656df9db 100644 --- a/frontend/lib/types/root.ts +++ b/frontend/lib/types/root.ts @@ -1,7 +1,7 @@ -import { z } from 'zod'; +import { z } from "zod"; export const rootModelSchema = z.object({ - id: z.string().uuid(), - created_at: z.date(), - updated_at: z.date() + id: z.string().uuid(), + created_at: z.date(), + updated_at: z.date(), }); diff --git a/frontend/lib/types/tag.ts b/frontend/lib/types/tag.ts index 2cc6d46d2..f748cc11b 100644 --- a/frontend/lib/types/tag.ts +++ b/frontend/lib/types/tag.ts @@ -3,7 +3,7 @@ import { z } from "zod"; import { rootModelSchema } from "../types/root"; export const tagSchemaIntermediate = z.object({ - name: z.string().max(255), + name: z.string().max(255), }); export const tagSchema = tagSchemaIntermediate.merge(rootModelSchema); diff --git a/frontend/lib/types/user.ts b/frontend/lib/types/user.ts index 80eb3181a..aedf4fa97 100644 --- a/frontend/lib/types/user.ts +++ b/frontend/lib/types/user.ts @@ -2,26 +2,26 @@ import { z } from "zod"; import { rootModelSchema } from "./root"; export const userSchemaIntermediate = z.object({ - role: z.enum(["super", "student"]), - first_name: z.string().min(1), - last_name: z.string().min(1), - email: z.string().email(), - college: z.string().optional(), - graduation_cycle: z.string().optional(), - graduation_year: z.number().optional(), - is_verified: z.boolean(), + role: z.enum(["super", "student"]), + first_name: z.string().min(1), + last_name: z.string().min(1), + email: z.string().email(), + college: z.string().optional(), + graduation_cycle: z.string().optional(), + graduation_year: z.number().optional(), + is_verified: z.boolean(), }); export const collegeSchema = z.enum([ - "CAMD", - "DMSB", - "KCCS", - "CE", - "BCHS", - "SL", - "CPS", - "CS", - "CSSH", + "CAMD", + "DMSB", + "KCCS", + "CE", + "BCHS", + "SL", + "CPS", + "CS", + "CSSH", ]); export type College = z.infer; @@ -29,4 +29,4 @@ export const yearSchema = z.enum(["1", "2", "3", "4", "5"]); export type Year = z.infer; export const userSchema = userSchemaIntermediate.merge(rootModelSchema); -export type User = z.infer; \ No newline at end of file +export type User = z.infer;