Skip to content

Commit 54d2f0d

Browse files
authored
Overhaul the job board to use a new API from reactibot (#335)
* Update netlify cli and functions * Remove 'verified email' gate, too finicky * Jobs api proxy * Fix checkboxes * Handle Discord rate limit error * Pass along all querystring values * Add @tanstack/react-query * Fix form bugs around unnecessary rerenders and blank initial values * Add a Reaction variant of Tag * Update job board page to sync with realtime data * oops logs * Nice cozy "Show More" behavior
1 parent 64e7b23 commit 54d2f0d

14 files changed

+2606
-3069
lines changed

netlify/functions/discordIdentity.mts

+10
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,16 @@ const handler = async (request: Request, context: Context) => {
1515
),
1616
]);
1717
const [user] = await Promise.all([userRes.json()]);
18+
if (!memberRes.ok) {
19+
if (memberRes.status === 429) {
20+
return new Response(
21+
JSON.stringify({
22+
message: memberRes.statusText,
23+
}),
24+
{ status: memberRes.status },
25+
);
26+
}
27+
}
1828
return new Response(
1929
JSON.stringify({
2030
user,

netlify/functions/jobs.mts

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
import type { Config, Context } from "@netlify/functions";
2+
3+
const handler = async (request: Request, context: Context) => {
4+
const { type } = context.params;
5+
const { searchParams } = new URL(request.url);
6+
7+
console.log(searchParams.toString());
8+
9+
return fetch(
10+
`https://api.reactiflux.com/jobs/${type}?${searchParams.toString()}`,
11+
{ headers: { "api-key": process.env.REACTIBOT_API_KEY || "" } },
12+
);
13+
};
14+
15+
export default handler;
16+
17+
export const config: Config = {
18+
path: "/api/jobs/:type",
19+
};

package.json

+5-4
Original file line numberDiff line numberDiff line change
@@ -21,8 +21,9 @@
2121
}
2222
},
2323
"dependencies": {
24-
"@netlify/functions": "^2.4.1",
24+
"@netlify/functions": "^3.0.0",
2525
"@next/third-parties": "^14.1.0",
26+
"@tanstack/react-query": "^5.65.0",
2627
"@types/node": "20.6.0",
2728
"@types/react": "17.0.37",
2829
"@types/styled-components": "^5.1.15",
@@ -37,14 +38,14 @@
3738
"mdast-util-to-string": "^3.1.0",
3839
"minireset.css": "^0.0.6",
3940
"moment": "^2.27.0",
40-
"netlify-cli": "^17.10.2",
41+
"netlify-cli": "^18.0.2",
4142
"next": "^12.3.4",
4243
"next-sitemap": "^4.2.2",
4344
"polished": "^3.6.5",
4445
"prettier": "^2.1.0",
4546
"pretty-quick": "^3.0.0",
46-
"react": "17.0.2",
47-
"react-dom": "17.0.2",
47+
"react": "~18",
48+
"react-dom": "~18",
4849
"react-headroom": "^2.1.2",
4950
"react-helmet": "^5.2.1",
5051
"react-hook-form": "^7.49.2",

src/components/DiscordAuth.tsx

+24-13
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,7 @@ type State =
3737
| "needsAuth"
3838
| "needsVerify"
3939
| "notMember"
40+
| "rateLimit"
4041
| "ok"
4142
| "err";
4243

@@ -52,15 +53,15 @@ const checkAuth = async (
5253
purgeToken();
5354
return "needsAuth";
5455
}
56+
if (res.status === 429) {
57+
return "rateLimit";
58+
}
5559
if (res.status !== 200) {
5660
return "err";
5761
}
5862

5963
const loadedUser = (await res.json()) as DiscordIdentity;
6064

61-
if (!loadedUser?.user?.verified) {
62-
return "needsVerify";
63-
}
6465
if (!loadedUser.isMember) {
6566
return "notMember";
6667
}
@@ -97,6 +98,16 @@ export const DiscordAuth = ({ children }: Props) => {
9798
<>
9899
{(() => {
99100
switch (state) {
101+
case "rateLimit":
102+
return (
103+
<div>
104+
<p>
105+
Oops! You got rate limited by Discord. Please try again in a
106+
minute or two.
107+
</p>
108+
</div>
109+
);
110+
100111
case "err":
101112
return (
102113
<div>
@@ -132,16 +143,16 @@ export const DiscordAuth = ({ children }: Props) => {
132143
💁
133144
</div>
134145
);
135-
case "needsVerify":
136-
return (
137-
<div>
138-
You don’t have a verified email associated with it. Please{" "}
139-
<a href="https://support.discord.com/hc/en-us/articles/213219267-Resending-Verification-Email">
140-
verify your email
141-
</a>{" "}
142-
and try again.
143-
</div>
144-
);
146+
// case "needsVerify":
147+
// return (
148+
// <div>
149+
// You don’t have a verified email associated with it. Please{" "}
150+
// <a href="https://support.discord.com/hc/en-us/articles/213219267-Resending-Verification-Email">
151+
// verify your email
152+
// </a>{" "}
153+
// and try again.
154+
// </div>
155+
// );
145156
case "needsAuth":
146157
default:
147158
return (

src/components/Form/Checkbox.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -66,7 +66,7 @@ export const Checkbox = ({ label, ...props }) => {
6666
const onChange = createChangeHandler(props);
6767
return (
6868
<p>
69-
<Label htmlFor={props.name}>
69+
<Label htmlFor={props.name} onClick={onChange}>
7070
<Checkmark hidden={!props.value}></Checkmark>
7171
<input
7272
checked={props.value}

src/components/Form/Form.js

+11-6
Original file line numberDiff line numberDiff line change
@@ -28,8 +28,9 @@ export const Form = React.forwardRef(function Form(
2828
) {
2929
const [status, setStatus] = React.useState(NONE);
3030
const [fieldState, setFieldState] = React.useState(
31-
origFields.reduce((acc, { defaultValue, name }) => {
32-
acc[name] = typeof defaultValue !== "undefined" ? defaultValue : "";
31+
origFields.reduce((acc, { defaultValue, value, name }) => {
32+
acc[name] =
33+
typeof defaultValue !== "undefined" ? defaultValue : value || "";
3334
return acc;
3435
}, {}),
3536
);
@@ -40,10 +41,14 @@ export const Form = React.forwardRef(function Form(
4041
// have to pull the value from the synthetic event here, because react throws it away
4142
const value = e.target.value;
4243
if (value !== fieldState[field.name]) {
43-
setFieldState((prev) => ({
44-
...prev,
45-
[field.name]: value,
46-
}));
44+
setFieldState((prev) => {
45+
const newState = {
46+
...prev,
47+
[field.name]: value,
48+
};
49+
onChange(newState);
50+
return newState;
51+
});
4752
}
4853
},
4954
value: fieldState[field.name],

src/components/JobSearch.js

-174
This file was deleted.

0 commit comments

Comments
 (0)