stevekrouse-docfeedbackform.web.val.run
Readme

Val Town Docs Feedback Form & Handler

This feedback form was linked on our docs site.

Screenshot 2023-09-07 at 14.24.25@2x.png

This val renders an HTML form, including pre-fills the user's email address if they've submitted the form in the past (via a cookie), and pre-fills the URL by grabbing it out of the query params.

It handles form submissions, including parsing the form, saving the data into @stevekrouse.docsFeedback, a private JSON val, and then returns a thank you message, and set's the user's email address as a cookie, to save them some keystrokes the next time they fill out the form.

Another val, @stevekrouse.formFeedbackAlert, polls on an interval for new form submissions, and if it finds any, forwards them on a private Val Town discord channel.

There are a number of subtleties to the way each of some features are implemented.

A user submitted three pieces of feedback in quick succession, so I thought it'd be nice if we remembered user's email addresses after their first form submissions. There are classically two ways to do this, cookies or localstorage. I choose cookies. It requires setting them in the response header and getting them out of the request header. I used a Deno library to parse the cookie but I set it manually because that seemed simpler.

You may be wondering about how I'm getting the referrer out of the query params instead of from the HTTP Referrer header. I tried that at first, but it's increasingly difficult to get path data from it due to more restrictive security policies. So instead I decided to include the URL data in a query param. I get it there via this script in my blog's site:

function updateFeedback(ref) { let feedback = [...document.getElementsByTagName('a')].find(e => e.innerText == 'Feedback') feedback.setAttribute('href', "https://stevekrouse-docfeedbackform.web.val.run/?ref=" + ref) } setTimeout(() => updateFeedback(document.location.href), 100); navigation.addEventListener('navigate', e => updateFeedback(e.destination.url));

Finally, you may be wondering why I queue up feedback in @stevekrouse.docsFeedback, a private JSON val, and then process it via @stevekrouse.formFeedbackAlert instead of sending it along to Discord directly in this val. I tried that originally but it felt too slow to wait for the API call to Discord before returning the "Thanks for your feedback" message. This is where the context.waitUntil method (that Cloudflare workers and Vercel Edge Functions support) would really come in handy – those allow you to return a Response, and then continue to compute. Currently Val Town requires you to stop all compute with the returning of your Response, so the only way to compute afterwards is to queue it up for another val to take over, and that's what I'm doing here.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
import { waterCSS } from "https://esm.town/v/stevekrouse/waterCSS";
import { html } from "https://esm.town/v/stevekrouse/html";
import { set } from "https://esm.town/v/std/set?v=11";
import { docsFeedback } from "https://esm.town/v/stevekrouse/docsFeedback";
export let docFeedbackForm = async (req: Request) => {
if (req.method === "POST") {
let formData = await req.formData();
docsFeedback.push({
time: Date.now(),
feedback: formData.get("feedback"),
url: formData.get("url"),
email: formData.get("email"),
});
await set(
"docsFeedback",
docsFeedback,
);
return html("Thanks for your feedback!", {
headers: {
"Set-Cookie": `email=${formData.get("email")}`,
},
});
}
const { getCookies } = await import("https://deno.land/std/http/cookie.ts");
let ref = new URL(req.url).searchParams.get("ref") ?? "";
let email = getCookies(req.headers).email ?? "";
return html(`${waterCSS}
<div style="margin-left: auto;margin-right: auto;width: 500px;padding: 30px;">
<h1>Val Town Docs Feedback</h1>
<form action="https://stevekrouse-docFeedbackForm.web.val.run" method="post">
<textarea name="feedback" placeholder="Feedback..." rows="4" cols="50"></textarea>
<br/><input name="url" value="${ref}" placeholder="URL" size="50" />
<br/><input name="email" value="${email}" placeholder="Email" size="30" />
<br/> <button>Submit</button>
</form>
<a href="https://www.val.town/v/stevekrouse.docFeedbackForm"><b><i>vt</i></b> source code</a>
</div>`);
};
Val Town is a social website to write and deploy JavaScript.
Build APIs and schedule functions from your browser.
Comments
Nobody has commented on this val yet: be the first!
October 23, 2023