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

Add playwright testing library + intial integration tests for the Join Form #58

Merged
merged 14 commits into from
Aug 1, 2024
27 changes: 27 additions & 0 deletions .github/workflows/playwright.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
name: Playwright Tests

Check failure on line 1 in .github/workflows/playwright.yml

View workflow job for this annotation

GitHub Actions / checkov-action

CKV2_GHA_1: "Ensure top-level permissions are not set to write-all"
Fixed Show fixed Hide fixed
on:
push:
branches: [ main, master ]
pull_request:
branches: [ main, master ]
jobs:
test:
timeout-minutes: 60
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-node@v4
with:
node-version: lts/*
- name: Install dependencies
run: npm ci
- name: Install Playwright Browsers
run: npx playwright install --with-deps
- name: Run Playwright tests
run: npx playwright test
- uses: actions/upload-artifact@v4
if: always()
with:
name: playwright-report
path: playwright-report/
retention-days: 30
6 changes: 5 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -40,4 +40,8 @@ yarn-error.log*
*.tsbuildinfo

# IDEs
.idea/
.idea/
/test-results/
/playwright-report/
/blob-report/
/playwright/.cache/
21 changes: 20 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -36,4 +36,23 @@ npm install
npm run dev
```

Finally, open `http://127.0.0.1:3000` in your web browser to interact with your copy of the application
Finally, open `http://127.0.0.1:3000` in your web browser to interact with your copy of the application

# Testing

We use `playwright` to do integration tests. You can run them with the following instructions:

1. Setup a dev instance of [meshdb](https://github.com/nycmeshnet/meshdb)

2. Copy `.env.sample` into `.env.local` and fill it out

3. Run the integration tests with `npx playwright test`

You can see what playwright is doing with `--headed`, and you can pause a test to
examine the browser by inserting `page.pause()` in your test.

To run a specific test, you can use `-g`:

`npx playwright test -g 'missing name'`

See the [docs](https://playwright.dev/docs/running-tests) for more information about playwright.
74 changes: 1 addition & 73 deletions app/api.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { z } from "zod";
import { JoinFormInput, JoinFormResponse, NNAssignFormInput, NNAssignFormResponse } from "@/app/io";

if (process.env.NEXT_PUBLIC_MESHDB_URL === undefined) {
throw new Error('Expected API url environment variable');
Expand All @@ -10,78 +10,6 @@ if (process.env.NEXT_PUBLIC_MESHDB_URL === undefined) {

const API_BASE = new URL(process.env.NEXT_PUBLIC_MESHDB_URL as string + "/api/v1/");

export const JoinFormInput = z.object({
first_name: z.string(),
last_name: z.string(),
email: z.string(),
phone: z.string(),
street_address: z.string(),
apartment: z.string(),
city: z.string(),
state: z.string(),
zip: z.number(),
roof_access: z.boolean(),
referral: z.string(),
ncl: z.boolean(),
})
export type JoinFormInput = z.infer<typeof JoinFormInput>

export const JoinFormResponse = z.object({
message: z.string().optional(),
building_id: z.number(),
member_id: z.number(),
install_number: z.number(),
member_exists: z.boolean(),
})
export type JoinFormResponse = z.infer<typeof JoinFormResponse>

export const NNAssignFormInput = z.object({
install_number: z.number(),
password: z.string(), // TODO: Salt/hash/whatever this
})
export type NNAssignFormInput = z.infer<typeof NNAssignFormInput>

export const NNAssignFormResponse = z.object({
message: z.string().optional(),
building_id: z.number(),
install_number: z.number(),
network_number: z.number(),
created: z.boolean(),
})
export type NNAssignFormResponse = z.infer<typeof NNAssignFormResponse>

export const QueryFormInput = z.object({
//route: z.string(),
legacy: z.string().optional(),
query_type: z.string(),
data: z.string(),
password: z.string(), // TODO: Salt/hash/whatever this
})
export type QueryFormInput = z.infer<typeof QueryFormInput>

export const QueryFormResponse = z.object({
count: z.number(),
next: z.string().nullable(),
previous: z.string().nullable(),
results: z.array(z.object({
install_number: z.number(),
street_address: z.string().nullable(),
unit: z.string(),
city: z.string(),
state: z.string(),
zip_code: z.string(),
name: z.string(),
phone_number: z.string().nullable(),
primary_email_address: z.string().nullable(),
stripe_email_address: z.string().nullable(),
additional_email_addresses: z.array(z.string()).nullable(),
notes: z.string(),
network_number: z.number().nullable(),
status: z.string(),
}))
})
export type QueryFormResponse = z.infer<typeof QueryFormResponse>

const get = async <S extends z.Schema>(url: string, schema: S, auth?: string, nextOptions?: NextFetchRequestConfig): Promise<ReturnType<S['parse']>> => {
const res = await fetch(new URL(url, API_BASE), {
headers: {
Expand Down
73 changes: 73 additions & 0 deletions app/io.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
import { z } from "zod";

export const JoinFormInput = z.object({
first_name: z.string(),
last_name: z.string(),
email: z.string(),
phone: z.string(),
street_address: z.string(),
apartment: z.string(),
city: z.string(),
state: z.string(),
zip: z.number(),
roof_access: z.boolean(),
referral: z.string(),
ncl: z.boolean(),
})
export type JoinFormInput = z.infer<typeof JoinFormInput>

export const JoinFormResponse = z.object({
message: z.string().optional(),
building_id: z.number(),
member_id: z.number(),
install_number: z.number(),
member_exists: z.boolean(),
})
export type JoinFormResponse = z.infer<typeof JoinFormResponse>

export const NNAssignFormInput = z.object({
install_number: z.number(),
password: z.string(), // TODO: Salt/hash/whatever this
})
export type NNAssignFormInput = z.infer<typeof NNAssignFormInput>

export const NNAssignFormResponse = z.object({
message: z.string().optional(),
building_id: z.number(),
install_number: z.number(),
network_number: z.number(),
created: z.boolean(),
})
export type NNAssignFormResponse = z.infer<typeof NNAssignFormResponse>

export const QueryFormInput = z.object({
//route: z.string(),
legacy: z.string().optional(),
query_type: z.string(),
data: z.string(),
password: z.string(), // TODO: Salt/hash/whatever this
})
export type QueryFormInput = z.infer<typeof QueryFormInput>

export const QueryFormResponse = z.object({
count: z.number(),
next: z.string().nullable(),
previous: z.string().nullable(),
results: z.array(z.object({
install_number: z.number(),
street_address: z.string().nullable(),
unit: z.string(),
city: z.string(),
state: z.string(),
zip_code: z.string(),
name: z.string(),
phone_number: z.string().nullable(),
primary_email_address: z.string().nullable(),
stripe_email_address: z.string().nullable(),
additional_email_addresses: z.array(z.string()).nullable(),
notes: z.string(),
network_number: z.number().nullable(),
status: z.string(),
}))
})
export type QueryFormResponse = z.infer<typeof QueryFormResponse>
8 changes: 5 additions & 3 deletions app/join/page.tsx
Original file line number Diff line number Diff line change
@@ -1,10 +1,12 @@
import { Footer } from "@/components/Footer/Footer";
import { Header } from "@/components/Header/Header";
import JoinForm from "@/components/JoinForm/JoinForm";
import Head from 'next/head';

// TODO:
// https://www.npmjs.com/package/react-phone-number-input
// https://www.npmjs.com/package/react-error-boundary
export const metadata = {
title: "Join Our Community Network!",
description: "Use this form to sign up for NYC Mesh",
};

export default async function Join() {
return <>
Expand Down
8 changes: 6 additions & 2 deletions components/JoinForm/JoinForm.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
"use client";

import { JoinFormInput, submitJoinForm } from "@/app/api";
import { submitJoinForm } from "@/app/api";
import { JoinFormInput } from "@/app/io";
import { recordJoinFormSubmissionToS3 } from "@/app/data";
import 'react-phone-number-input/style.css';
import PhoneInput from 'react-phone-number-input/input';
Expand Down Expand Up @@ -121,7 +122,7 @@ const JoinForm = () => {
<div className={styles.block}>
<h3>Address Info</h3>
<input type="text" name="street_address" placeholder="Street Address" required />
<input type="text" name="apartment" placeholder="Unit #" required />
<input type="text" name="apartment" placeholder="Unit #" />
WillNilges marked this conversation as resolved.
Show resolved Hide resolved
<input type="text" name="city" placeholder="City" required />
<Select name="state" placeholder="State" options={options} defaultValue={options[0]} className={styles.drop} required />
<input type="number" name="zip" placeholder="Zip Code" required />
Expand All @@ -143,13 +144,16 @@ const JoinForm = () => {
variant="contained"
size="large"
sx={{ width: "12rem", fontSize: "1rem", m:"1rem"}}
name="submit_join_form"
>
{ isLoading ? "Loading..." : (submitted ? "Thanks!" : "Submit") }
</Button>
</div>
</form>
</div>
<div className="toasty">
<ToastContainer />
</div>
</>
}

Expand Down
64 changes: 64 additions & 0 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@
"zod": "^3.22.4"
},
"devDependencies": {
"@playwright/test": "^1.45.3",
"@types/react": "^18.2.48"
}
}
Loading
Loading