Daniel Goldelman
This project is a web application that has a few core goals:
- A user is able to access the website, which has the following core features:
- A button for the user to click.
- A number indicating how many times the button has been clicked.
- This data is stored in a persistent database.
- This web application is hosted in a public repository with CI/CD.
CI/CD Requirement:
- Application hosted on Vercel
- Uploads the latest application to Vercel upon new code push (from main branch of Github repository)
Accomplished Personalized Requirements:
- Keep track of how many times a unique individual has pressed the button.
- Add monitoring and logging so that admins can audit who has pressed the button and when.
- Dashboard that compares the number of clicks by all usernames in the database.
- README.md
- Form for submitting a username
- Users cannot click the button to edit the database until the username field has been filled out
- Anyone can see how many times the button has been clicked, irrespective of whether or not they are signed in
- Once a user has provided a username, they can see how many times someone with that username has clicked the button
- There exists an admin page, accessible by using the string "admin" as the username on the homepage.
- Once on the admin page, an admin can choose between two views:
- Bar Graph:
- Admins can see how many times each user has clicked the button, ordered by time the username's first click was registered in the database
- Search:
- Enter in a username and click "submit"
- If that username has been used to click the button, users will see:
- Text describing how many times the username has been used to click the button
- Scrollable list of when the button was clicked (date and time)
- If that username has not been used to click the button:
- Tells the admin that the provided username has not been used to click the button
- If that username has been used to click the button, users will see:
- Enter in a username and click "submit"
- Bar Graph:
- Once on the admin page, an admin can choose between two views:
- This project was bootstrapped using
create-next-app
from next.js. This method provides options, enumerated here:- What is your project named? -> goldelman_ayr_take_home_project
- Would you like to use TypeScript? -> Yes
- Would you like to use ESLint? -> Yes
- Would you like to use Tailwind CSS? -> Yes
- Would you like to use
src/
directory? -> No - Would you like to use App Router? (recommended) -> Yes
- Would you like to customize the default import alias (@/*)? -> Yes
- Frontend:
React.js
frontend for component-based renderingnext.js
utilized throughout for rendering and routingdotenv
used for absorbing secrets in.env
files, not provided publically for protection, and for seeding databasetailwindcss
used for stylingeslint
for linting and formattingchart.js
for generating the graph in the admin graph page- Written in Typescript
- Client-side rendering
- Backend queried using
fetch
- Backend
node
backend used for routing and querying the database@vercel/postgres
used to create SQL queries to interact with the database- Written in Typescript and SQL
- Server-side functions for secure interaction with the database
- Database
- Postgres
- Hosted on Vercel
.env
database secrets hosted alongside within Vercel
api/global
:- GET(request):
- Parameters From Headers: None
- Returns: Number of times all usernames have clicked the button
- GET(request):
api/userGiven
:- GET(request):
- Parameters From Headers:
- username: string representation of a username
- Returns: Number of times that username has been used to click the button
- Parameters From Headers:
- POST(request):
- Parameters From Headers:
- username: string representation of a username
- time: exact timestamp of when the button was clicked (in milliseconds)
- Returns: Confirmation message
- Parameters From Headers:
- GET(request):
api/admin
:- GET(request):
- Parameters From Headers:
- username: string representation of a username
- Returns: Tuple(number of times this username was used to click the button, list of times that the username was used to click the button)
- Parameters From Headers:
- GET(request):
users
table columns:- id: UUIDv4
- username: string representation of username
- time: timestamp of when the button was clicked
- I have experience building websites in Vercel/Next.js, so I decided to use their service for this project. More information on how this affects performance is in the related section.
- All colors begining with
ayr-*
described intailwind.config.ts
were taken directly from the AYR website using a color picker Chrome extension. - Process for updating the frontend with data:
fetch
requests were sent from client-side components to the server- Server-side components handle sql queries to the database
vercel/postgress
handles all sql queries, so there is protection against sql injection (information via their docs)
- Data forwarded to client-side components
- Frontend updated
- node v21.7.1+
- npx v10.5.0+
- Install node v21.7.1
- Install npx 10.5.0
- Create an account on Github
- Install Git or the Github CLI
- cd into the directory you would like to place this directory into
- Run
git clone
+ {url you are visiting this from} - cd into the directory
- Run
npm install
to install dependancies - On Vercel:
- Log in
- Nagigate to the project related to this repo
- Navigate to the storage tab
- Navigate to "goldelman-ayr-postgres"
- Click on ".env.local" in the list
- Copy the snippet
- Back in the directory
- Create a new file named ".env.local"
- Paste the secrets obtained in vercel
- Run
npm run dev
to start a development version of this web app - In a web browser, enter "localhost:3000" or "127.0.0.1:3000"
Instructions for Reproduction (If you do not have access to the Vercel account used for production build)
Replace step 9 with an email to me at daniel.goldelman@gmail.com, and I will send you the ".env.local" file
Due to limitations with the Hobby plan of Vercel, the database is only hosted on one server in one location (Washington, DC) and the connection is quite slow since it is rate-limited for non-paying projects. This means that reliability, availability, and performance are all based on current usage of the server, and that this project is low on the priority lists for the Vercel service. This limited implementation also means that there are limits to the amount of data able to be stored or computed monthly. Improved reliability, availability, and performace could easily be achieved by choosing a paying service for hosting the database, or alternatively to move the entire backend (including the APIs and database) to some paid service, Vercel or otherwise.
There is are two obvious security flaws in the current system due to lack of authentication. First, anyone can sign in as any user without a password. Second, anyone can access the admin page, which would normally be much more secure. There are many options for how to introduce authentication (NextAuth, Auth0, Firebase, Okta, etc.), but this was outside of the bounds of the current implementation.
Due to Vercel's efforts, I do not believe there are significant security risks related to SQL injection (mentioned above) or improper access to the database (since all requests operate on the server and are protected and excluded from public repositories).
- The number of times the button has been clicked automatically updates for multiple users who are accessing the site at the same time, better if live updating. Response: I would achieve this by setting up an exchange from the server to each client currently accessing the site, sending them an updated value. This would necessitate keeping a list of current users on the site.
- Create an authentication system where users have to login to press the button. Response: I would use NextAuth to obtain the user's login, which would provide me an ID or email which I would then use as an authentic token, then ask the user for their desired username.
- Add a dashboard that shows a graph of total clicks over time and/or clicks per minute. I already have the ability to intake data from the server and then graph it, so adjusting what the graph shows is a fairly straightforward algorithmic process.
- For total clicks over time, I would break it down by day/hour/minute, based on a dropdown menu. The time is stored as milliseconds from
Date.now()
, and then converted into date-time format using theDate
functionality from Javascript/Typescript. Funneling each time into buckets is trivial from there. - For clicks per minute, the process is similar. I could provide some format of entry form where the user could specify what times they would like to look at, and then filter the millisecond times into buckets.
- For total clicks over time, I would break it down by day/hour/minute, based on a dropdown menu. The time is stored as milliseconds from
- / (homepage)
- Header containing the text "Click Counter", subtext "AYR Take Home Project"
- Form where a user can enter a username, then click submit (will increase in size on hover)
- Before Submit button has been clicked:
- Text that reads: "Enter a username, then click the button!"
- Red Button that reads: "Click Me". If clicked:
- Alert shown reading "Enter a username before clicking the button!"
- If Submit has been clicked without a provided username
- Dropdown that reads "Please fill out this field", the default action for required elements (on chrome)
- If Submit has been clicked with a provided username:
- if the username is "admin":
- User is routed to "/admin"
- else:
- Text that reads: "Hello, {username}!"
- Text that reads: "Your Clicks: {number of times the provided username has been used to click the button}"
- Blue Button that reads: "Click Me"
- When the button is clicked:
- The button turns Green and the text turns black
- Number of clicks should update (personal and global)
- When the button is clicked:
- if the username is "admin":
- Before Submit button has been clicked:
- Footer containing the text "Website by Daniel Goldelman"
- /admin (admin page)
- Header containing the text "Admin Page", subtext "AYR Take Home Project"
- Blue Button with the text "Go To Bar Graph". When clicked:
- Will turn Green with Black text
- Will route to "/admin/barGraph"
- Blue Button with the text "Go To Search"
- Will turn Green with Black text
- Will route to "/admin/barGraph"
- Blue Button with the text "Back To Homepage"
- Will turn Green with Black text
- Will route to "/"
- Footer containing the text "Website by Daniel Goldelman"
- /admin/barGraph
- Header containing the text "Admin Page", subtext "AYR Take Home Project"
- Blue Button with the text "Back To Admin Page"
- Will turn Green with Black text
- Will route to "/admin"
- Graph comparing all usernames' number of clicks, in order of who clicked first
- Footer containing the text "Website by Daniel Goldelman"
- /admin/search
- Header containing the text "Admin Page", subtext "AYR Take Home Project"
- Blue Button with the text "Back To Admin Page"
- Will turn Green with Black text
- Will route to "/admin"
- Form where a user can enter a username, then click submit (will increase in size on hover)
- If Submit has been clicked without a provided username:
- Dropdown that reads "Please fill out this field", the default action for required elements (on chrome)
- If Submit has been clicked with a provided username:
- If the provided username has clicked the button:
- Text reading: "User: {username}"
- Text reading: "Clicks: {number of times this username has been used to click the button}
- Scrollable list of all the dates and times the button was clicked using this username
- Format: month/day/year, hour:minute:second {AM|PM}
- If the provided username has clicked the button:
- If Submit has been clicked without a provided username:
- Footer containing the text "Website by Daniel Goldelman"