mirror of
				https://github.com/monkeytypegame/monkeytype.git
				synced 2025-10-26 16:57:07 +08:00 
			
		
		
		
	* Fixed typo
* Created method for adding theme in the UserDAO:
* Created function for checking if custom theme object is valid
* Exported the isThemeValid function
* Added controller for adding customTheme
* Created route for adding custom theme
* Created rateLimit for adding customTheme
* Fixed typo
* Fixed incorrect color length
* Added method for removing, getting and editing custom themes on the backend
* Moved validations from controllers to routes and some aesthetic changes in the user dao
* Started working on frontend and some minor changes in the backend
- Commandline support for custom themes
- Allow user to shift to their custom theme using Shift-click
- Updated the backend to be compatible with some changes
- Create a new custom theme for users with old system to prevent their custom theme loss
* Fixed custom theme type in ape and now new custom theme is created if user clicks the custom tab and doesn't already have one
* Fixed ape type issue
* Format html file
* Fixed wrong tab being active
* Created new custom theme edit section
* Fixed bug where user theme would have impact on icons with custom theme
* Update customThemes API
* Updated the custom theme sharing option to work with multiple custom themes
* Started working on the UI for custom theme buttons
* Added DOM event for clicking custom theme buttons
* Updated the updateActiveButton to work with multiple custom themes
* Removed favorite button for themes and fixed bug where double theme buttons were being added
* Fixed bug where preset theme buttons were not appearing if user has applied custom theme on website load and refreshed
* Moved DOM event for sharing custom theme to more appropriate place
* Integrated the save custom theme button with the changes
* Fixed bug with custom theme tab buttons and theme buttons
* Fixed commented div
* Replaced 'sds' with a meaningful message for custom theme buttons
* Integrated the delete button for deleting custom themes and fixed bug where id of newly added custom theme was not set properly
* Integrated the add button and name field for custom themes editing
* Added addCustomThemeWrapper element
I added it before but seems like vscode and other editors can't handle large files
* Removed some debug statements
* Removed some more debug statements
* Used parial types for custom theme. Thanks Bruce
* Removed unnecssary try catch blocks. Thanks Bruce
* Rephrased custom theme API messages
* Set new theme fields explicitly to prevent validtion failures and rephrased API message
* Replaced let with const
* Replaced let with const for _id
* Replaced let with const and used nullish coalescing
* Improved code quality in User DAO
* Strict equality in user DAO
* Moved validation scheme to a variables at the top of file
* Fixed bugs with strict equality checks
* Renamed themeId to themeID for consistency
* Made customThemes a required type in db to remove unnecessary undefined checks
* Uncommented GET API endpoint
* Prevent colorId being updated on custom theme name chnage
* Removed debug log
* Added loader on api calls
* Commenced shift from customThemeIndex to customThemeId
* Added required to themeColors schema
* Temp fix for validation fail for customThemeId
* Changed default value of customThemeId back to ''
* Temp fix for validation fail for customThemeId
* Fixed minor bug
* Fixed bug where account-controller would pass undefined to ThemeController.set
* Created methods in db.ts for adding, deleting and editing custom themes. Created new interface for raw custom themes and renamed ape methods
* Removed repeating code in account-controller
* Removed repeating code in theme-picker
* Removed setThemes in config
* Fixed minor bug
* Removed repeating code in user DAO
* Made custom themes available to registered users only
* Fixed minor bug
* Removed debug log and updated custom theme commands before showing list
* Added popup for confirming custom theme deletion
* Added custom option for random theme
* Minor improvement
* Workaround for local config firing before firebase initalization
* Removed debug log and created workaround for migration
* Added legacy customTheme config field
* Replaced workaround
* Changed put to patch
* Changed put to patch
* Added customTheme field back
* Integrated customTheme into to feature
* Added notifications for users when they access custom theme cmd option without being logged in
* Removed debug logs and comments
* Replaced literals with constant. Thanks Bruception
* Fixed wrong querySelector parameters and reset custom theme colors after deleting a custom theme
* added notification on save
* duplicating object instead of referencing
* Added return type on function
* Fixed wrong notification code
* spreading default config instead of referencing
* added index, psas, configs, presets
* camel_case
* added ape keys, leaderboards, results, quotes
* Modified setCustomTheme
* Modified setCustomThemeId
* Added tip for random themes settings
* Modified setCustomThemeId
* Now load custom theme before account loading
* Added custom theme compatibility for non-logged in users to theme-controller
* Now update tabs and buttons on custom theme config value change and modified boolean checks to use customTheme instead of customThemeId
* Fixed bug
* Refactoring in theme-controller.ts
* Enable custom theme support in commandline for logged out users
* More refactoring in theme-controller.ts
* Added custom theme compatibility for logged out users
* Removed double events in settings.ts and now turn on custom theme upon applying
* Fixed bug and recursive call
* Fixed bug
* Fixed random theme custom option
* Fix jquery wrong syntax
* Readded notification upon custom theme edit
* One notification upon error only
* Change notification type
* New custom themes now have default colors
* Notification on custom theme edit for non-logged in users
* Refresh buttons upon settings load
* missing gitignore
* updated message
* updated message
* setting config to unchanged when logging in to avoid issues with applying db config
* reverted some over complicated code, excessive auth checks
* removed customthemeid from config
* not setting custom theme id
* removed all customthemeid references
* removed commented code
* removed name field
* added edit button
* unused file
* removed popup
* removed add button, removed text
* removed duplicate code
* added simple popup checkbox support
* whitespace
* added custom theme popups
* removed warning when no custom themes were found
* removed add button click handler
* added function to save custom theme
* saving current theme not default
* removed custom theme id from default config
* not creating new theme by default, just applying
* reacting to customThemeColors save
* unnecessary function call
* removed unused code
* small refactor
* spacing
* unnecessary code
* turned off warnings for non null asertion
* showing theme name when randomising customs
* Revert "turned off warnings for non null asertion"
This reverts commit 433e1dc767.
* optional with default instead
* fixed custom theme colors always loaded on page load
* fixed custom theme buttons not showing up
* fixed various loading issues
* fixed custom theme edit styles
* showing custom in footer, removed unused code
* savaing custom theme colors
fixed typos
* changing theme
* updated custom theme buttons styling
* scaling custom theme buttons on hover
* not updating settings on theme event
* fixed quote id
* only showing custom themes when logged in
* updating save button text depending on auth state
* fixed double notification when trying to save too many custom themes
* fixed custom theme saving when signed out
* removed user check from db
* fixed exception when signed out user tried to open the custom themes command line
* ignoring file when compiling
* typo
* avoiding href errors
* setting href to an existing file
this fixes firefox custom themes not working
* better hex color regex
* spacing
* renamed function
* typo
* destructuring request
* removed unused function
* removed unused code
* removed unused code
* type fix
* removed non capturing group
* saving colors to config before saving custom theme
* encoding in base64
* added handler that can load themes in the old and new format from the url
Co-authored-by: Rizwan Mustafa <rizwanmustafa0000@gmail.com>
Co-authored-by: Rizwan Mustafa <69350358+rizwanmustafa@users.noreply.github.com>
		
	
			
		
			
				
	
	
		
			326 lines
		
	
	
	
		
			7.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			326 lines
		
	
	
	
		
			7.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { Response, NextFunction } from "express";
 | |
| import rateLimit, { Options } from "express-rate-limit";
 | |
| import MonkeyError from "../utils/error";
 | |
| 
 | |
| const REQUEST_MULTIPLIER = process.env.MODE === "dev" ? 100 : 1;
 | |
| 
 | |
| const getAddress = (req: MonkeyTypes.Request, _res: Response): string => {
 | |
|   return (req.headers["cf-connecting-ip"] ||
 | |
|     req.headers["x-forwarded-for"] ||
 | |
|     req.ip ||
 | |
|     "255.255.255.255") as string;
 | |
| };
 | |
| 
 | |
| const customHandler = (
 | |
|   _req: MonkeyTypes.Request,
 | |
|   _res: Response,
 | |
|   _next: NextFunction,
 | |
|   _options: Options
 | |
| ): void => {
 | |
|   throw new MonkeyError(429, "Too many attempts, please try again later.");
 | |
| };
 | |
| 
 | |
| const ONE_HOUR = 1000 * 60 * 60;
 | |
| 
 | |
| // Config Routing
 | |
| export const configUpdate = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 500 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const configGet = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 120 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| // Leaderboards Routing
 | |
| export const leaderboardsGet = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| // New Quotes Routing
 | |
| export const newQuotesGet = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 500 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const newQuotesAdd = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const newQuotesAction = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 500 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| // Quote Ratings Routing
 | |
| export const quoteRatingsGet = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 500 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const quoteRatingsSubmit = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 500 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| // Quote reporting
 | |
| export const quoteReportSubmit = rateLimit({
 | |
|   windowMs: 30 * 60 * 1000, // 30 min
 | |
|   max: 50 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| // Presets Routing
 | |
| export const presetsGet = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const presetsAdd = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const presetsRemove = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const presetsEdit = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| // PSA (Public Service Announcement) Routing
 | |
| export const psaGet = rateLimit({
 | |
|   windowMs: 60 * 1000,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| // Results Routing
 | |
| export const resultsGet = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const resultsAdd = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 500 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const resultsTagsUpdate = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 30 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const resultsDeleteAll = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 10 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const resultsLeaderboardGet = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const resultsLeaderboardQualificationGet = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| // Users Routing
 | |
| export const userGet = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userSignup = rateLimit({
 | |
|   windowMs: 24 * ONE_HOUR, // 1 day
 | |
|   max: 3 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userDelete = rateLimit({
 | |
|   windowMs: 24 * ONE_HOUR, // 1 day
 | |
|   max: 3 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userCheckName = rateLimit({
 | |
|   windowMs: 60 * 1000,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userUpdateName = rateLimit({
 | |
|   windowMs: 24 * ONE_HOUR, // 1 day
 | |
|   max: 3 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userUpdateLBMemory = rateLimit({
 | |
|   windowMs: 60 * 1000,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userUpdateEmail = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userClearPB = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userTagsGet = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userTagsRemove = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 30 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userTagsClearPB = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 60 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userTagsEdit = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 30 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userTagsAdd = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 30 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userCustomThemeGet = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 30 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userCustomThemeAdd = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 30 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userCustomThemeRemove = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 30 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userCustomThemeEdit = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 30 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const userDiscordLink = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 15 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const usersTagsEdit = userDiscordLink;
 | |
| 
 | |
| export const userDiscordUnlink = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 15 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| // ApeKeys Routing
 | |
| export const apeKeysGet = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 120 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const apeKeysGenerate = rateLimit({
 | |
|   windowMs: ONE_HOUR,
 | |
|   max: 15 * REQUEST_MULTIPLIER,
 | |
|   keyGenerator: getAddress,
 | |
|   handler: customHandler,
 | |
| });
 | |
| 
 | |
| export const apeKeysUpdate = apeKeysGenerate;
 | |
| 
 | |
| export const apeKeysDelete = apeKeysGenerate;
 |