Compare commits
	
		
			183 Commits
		
	
	
		
			poise-2
			...
			06e1474396
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 06e1474396 | |||
|  | adb9c728f4 | ||
|  | fc02eaea4a | ||
|  | 91f87302fb | ||
|  | 97f186dc33 | ||
|  | 6eaa6f0f28 | ||
|  | 9db0fa2513 | ||
|  | ca13fd4fa7 | ||
|  | 55acc8fd16 | ||
|  | 145711fa5d | ||
|  | 5524215786 | ||
|  | e8bd05893f | ||
|  | e3d3418f99 | ||
|  | 2681280a39 | ||
|  | 00579428a1 | ||
|  | b8ef999710 | ||
|  | e8f84e281a | ||
|  | 8ddff698e5 | ||
|  | 541633270c | ||
|  | 25286da5e0 | ||
|  | 4bad1324b9 | ||
|  | bd1462a00c | ||
|  | 56ffc43616 | ||
|  | 52cf642455 | ||
|  | 0bf578357a | ||
|  | 6e9eccb62e | ||
|  | 6ea28284ce | ||
|  | a6525f3052 | ||
|  | 348639270d | ||
|  | 37177c2431 | ||
|  | 8587bed703 | ||
|  | 6c9af1ae8e | ||
|  | 7695b7a476 | ||
| 651da7b28e | |||
| eb086146bf | |||
| 4ebd705e5e | |||
| 5a85f1d83a | |||
| 68ba25886a | |||
|  | e25bf6b828 | ||
|  | 5a386daa9d | ||
|  | 0d4a02fb1e | ||
|  | e135a74a9b | ||
|  | 77f17c8dc2 | ||
|  | 6a94f990cf | ||
|  | 3aa5bd37aa | ||
|  | fa83fed1af | ||
|  | 666cb7fa2f | ||
|  | a5678e15dc | ||
|  | 9405cfcee9 | ||
|  | cb25d02cdf | ||
|  | bfe651a125 | ||
|  | dc5e52d9ce | ||
|  | 229ada83e1 | ||
|  | 13171d6744 | ||
|  | 2ad941c94c | ||
|  | 924d31e978 | ||
|  | f9a1b23212 | ||
|  | ae5795a7ea | ||
|  | ee36c38eda | ||
|  | eca7df3d9f | ||
|  | 902b7e1b4a | ||
|  | db1a53a797 | ||
|  | 3605d71b73 | ||
|  | ea2cea573e | ||
|  | d5fa8036e8 | ||
|  | b8707bbc9a | ||
|  | 99eea16f62 | ||
|  | 88737302f3 | ||
|  | 213e3a5100 | ||
|  | 8fa1402ecc | ||
|  | e63996bb61 | ||
|  | 9ede879630 | ||
|  | 88e9826a62 | ||
|  | 5d655c7e6d | ||
|  | 51c9d8a7ae | ||
|  | 90df265114 | ||
|  | e65429aa9c | ||
|  | 8d2232f0da | ||
|  | a58b9866ea | ||
|  | b1f25be5d7 | ||
|  | f0f9787326 | ||
|  | 302f5835e6 | ||
|  | 58c778632e | ||
|  | 5671fd462b | ||
|  | 5ac9733f15 | ||
|  | 01dc0334fd | ||
|  | 4a17aac15c | ||
|  | 8ce4fc9c6d | ||
|  | b4f07cfc1c | ||
|  | 8799089b2d | ||
|  | 88c4830209 | ||
|  | 4dd3df5cc2 | ||
|  | 369a325a46 | ||
|  | 1a1a0fdefb | ||
|  | dda8bd3e10 | ||
|  | edbfc92cb9 | ||
|  | 6de11f09db | ||
|  | 284bfcd9ad | ||
|  | 3d627b5bf0 | ||
|  | c3c0dbbbae | ||
|  | 64dd81e941 | ||
|  | 799298ca34 | ||
|  | fa542bb24f | ||
|  | e025d945cf | ||
|  | bb1c61d0b9 | ||
|  | 1519474f93 | ||
|  | 9d8622f418 | ||
|  | a66db37b33 | ||
|  | c8c1a171d4 | ||
|  | 88cfb829e3 | ||
|  | 16be7a328e | ||
|  | 04babf7930 | ||
|  | 96bc09e8b5 | ||
|  | 976fb91ecc | ||
|  | 1305b6e64e | ||
|  | cdfe44d958 | ||
|  | c824a36832 | ||
|  | c4bd2c1d18 | ||
|  | 561555ab7e | ||
|  | 115fbd44cb | ||
|  | aa931328b0 | ||
| 4b42966284 | |||
| 523ab7f03a | |||
| 6e831c8253 | |||
|  | 4416e5d175 | ||
|  | 734a39a001 | ||
|  | 98191d29ee | ||
|  | 1c4c4a8b31 | ||
|  | d496c81003 | ||
|  | 094d210f64 | ||
|  | 314c72e132 | ||
|  | 4e0163f2cb | ||
|  | e5b8c418af | ||
|  | 3ef8584189 | ||
|  | df2ad09c86 | ||
|  | d70fb24eb1 | ||
|  | 3150c7267d | ||
|  | 6e65e4ff3d | ||
|  | 67a4db2e9a | ||
|  | e9bcb1973f | ||
|  | 9b87fd4258 | ||
|  | a49a849917 | ||
|  | aa74a7f9a3 | ||
|  | 08e4c6cb57 | ||
|  | 6e087bd2dd | ||
| e9792e6322 | |||
| 130504b964 | |||
| 2a8117d0c1 | |||
| 94bfd39085 | |||
| 40cd5f8a36 | |||
| 133b00a2ce | |||
| 57336f5c81 | |||
| b62d24c024 | |||
| 8f8235a86e | |||
| c8f646a8fa | |||
| ecaa382a1e | |||
| 8991198fd3 | |||
|  | f20b95a482 | ||
|  | 8dd7dc6409 | ||
|  | c799d10727 | ||
|  | ceb6fb7b12 | ||
|  | 6708abdb0f | ||
|  | a38f6024c1 | ||
|  | 7d8748e3ef | ||
|  | bb3386c4e8 | ||
|  | 25b84880a5 | ||
|  | 7b6e967a5d | ||
|  | 2781f2923e | ||
|  | 03f08f0a18 | ||
|  | 79c86d43f2 | ||
|  | e19af54caf | ||
|  | f4213c6a83 | ||
|  | f56db14720 | ||
|  | 6f7d0f67b3 | ||
|  | bfc2d71ca0 | ||
|  | 8eb46f1f23 | ||
|  | c4087bf569 | ||
|  | f25cfed8d7 | ||
|  | d2a8bd1982 | ||
|  | 437ee6b446 | ||
|  | 7d43aa5918 | ||
|  | 8bad95510d | ||
|  | d7a0b727fb | 
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -2,6 +2,6 @@ | |||||||
| .env | .env | ||||||
| /venv | /venv | ||||||
| .cargo | .cargo | ||||||
| assets |  | ||||||
| out.json |  | ||||||
| /.idea | /.idea | ||||||
|  | web/static/index.html | ||||||
|  | web/static/assets | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | [submodule "reminder-dashboard"] | ||||||
|  | 	path = reminder-dashboard | ||||||
|  | 	url = gitea@gitea.jellypro.xyz:jude/reminder-dashboard | ||||||
							
								
								
									
										2655
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
							
								
								
									
										51
									
								
								Cargo.toml
									
									
									
									
									
								
							
							
						
						| @@ -1,32 +1,59 @@ | |||||||
| [package] | [package] | ||||||
| name = "reminder_rs" | name = "reminder-rs" | ||||||
| version = "1.6.0-beta3" | version = "1.6.48" | ||||||
| authors = ["jellywx <judesouthworth@pm.me>"] | authors = ["Jude Southworth <judesouthworth@pm.me>"] | ||||||
| edition = "2018" | edition = "2021" | ||||||
|  | license = "AGPL-3.0 only" | ||||||
|  | description = "Reminder Bot for Discord, now in Rust" | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| poise = "0.2" | poise = "0.5" | ||||||
| dotenv = "0.15" | dotenv = "0.15" | ||||||
| tokio = { version = "1", features = ["process", "full"] } | tokio = { version = "1", features = ["process", "full"] } | ||||||
| reqwest = "0.11" | reqwest = "0.11" | ||||||
| regex = "1.4" | lazy-regex = "3.0.2" | ||||||
|  | regex = "1.9" | ||||||
| log = "0.4" | log = "0.4" | ||||||
| env_logger = "0.8" | env_logger = "0.10" | ||||||
| chrono = "0.4" | chrono = "0.4" | ||||||
| chrono-tz = { version = "0.5", features = ["serde"] } | chrono-tz = { version = "0.8", features = ["serde"] } | ||||||
| lazy_static = "1.4" | lazy_static = "1.4" | ||||||
| num-integer = "0.1" | num-integer = "0.1" | ||||||
| serde = "1.0" | serde = "1.0" | ||||||
| serde_json = "1.0" | serde_json = "1.0" | ||||||
| serde_repr = "0.1" | serde_repr = "0.1" | ||||||
| rmp-serde = "0.15" | rmp-serde = "1.1" | ||||||
| rand = "0.7" | rand = "0.8" | ||||||
| levenshtein = "1.0" | levenshtein = "1.0" | ||||||
| sqlx = { version = "0.5", features = ["runtime-tokio-rustls", "macros", "mysql", "bigdecimal", "chrono"]} | sqlx = { version = "0.7", features = ["runtime-tokio-rustls", "macros", "mysql", "bigdecimal", "chrono", "migrate"]} | ||||||
| base64 = "0.13.0" | base64 = "0.21.0" | ||||||
|  |  | ||||||
| [dependencies.postman] | [dependencies.postman] | ||||||
| path = "postman" | path = "postman" | ||||||
|  |  | ||||||
| [dependencies.reminder_web] | [dependencies.reminder_web] | ||||||
| path = "web" | path = "web" | ||||||
|  |  | ||||||
|  | [package.metadata.deb] | ||||||
|  | depends = "$auto, python3-dateparser (>= 1.0.0)" | ||||||
|  | suggests = "mysql-server-8.0, nginx" | ||||||
|  | maintainer-scripts = "debian" | ||||||
|  | assets = [ | ||||||
|  |     ["target/release/reminder-rs", "usr/bin/reminder-rs", "755"], | ||||||
|  |     ["conf/default.env", "etc/reminder-rs/config.env", "600"], | ||||||
|  |     ["conf/Rocket.toml", "etc/reminder-rs/Rocket.toml", "600"], | ||||||
|  |     ["web/static/**/*", "lib/reminder-rs/static", "644"], | ||||||
|  |     ["reminder-dashboard/dist/static/**/*", "lib/reminder-rs/static", "644"], | ||||||
|  |     ["web/templates/**/*", "lib/reminder-rs/templates", "644"], | ||||||
|  |     ["healthcheck", "lib/reminder-rs/healthcheck", "755"], | ||||||
|  |     ["cron.d/reminder_health", "etc/cron.d/reminder_health", "644"], | ||||||
|  | #    ["nginx/reminder-rs", "etc/nginx/sites-available/reminder-rs", "755"] | ||||||
|  | ] | ||||||
|  | conf-files = [ | ||||||
|  |     "/etc/reminder-rs/config.env", | ||||||
|  |     "/etc/reminder-rs/Rocket.toml", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | [package.metadata.deb.systemd-units] | ||||||
|  | unit-scripts = "systemd" | ||||||
|  | start = false | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								Containerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | |||||||
|  | FROM ubuntu:20.04 | ||||||
|  |  | ||||||
|  | ENV RUSTUP_HOME=/usr/local/rustup \ | ||||||
|  |     CARGO_HOME=/usr/local/cargo \ | ||||||
|  |     PATH=/usr/local/cargo/bin:$PATH | ||||||
|  |  | ||||||
|  | RUN apt update && DEBIAN_FRONTEND=noninteractive TZ=Etc/UTC apt install -y gcc gcc-multilib cmake pkg-config libssl-dev curl mysql-client-8.0 | ||||||
|  | RUN curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y --no-modify-path --profile minimal --default-toolchain nightly | ||||||
|  | RUN cargo install cargo-deb | ||||||
							
								
								
									
										45
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -2,23 +2,41 @@ | |||||||
| Reminder Bot for Discord. | Reminder Bot for Discord. | ||||||
|  |  | ||||||
| ## How do I use it? | ## How do I use it? | ||||||
| We offer a hosted version of the bot. You can invite it with: **https://invite.reminder-bot.com**. The catch is that repeating  | I offer a hosted version of the bot. You can invite it with: **https://invite.reminder-bot.com**. The catch is that repeating  | ||||||
| reminders are paid on the hosted version of the bot. Keep reading if you want to host it yourself. | reminders are paid on the hosted version of the bot. Keep reading if you want to host it yourself. | ||||||
|  |  | ||||||
| You'll need rustc and cargo for compilation. To run, you'll need Python 3 still (due to no suitable replacement for dateparser in Rust) | You'll need rustc and cargo for compilation. To run, you'll need Python 3 still (due to no suitable replacement for dateparser in Rust) | ||||||
|  |  | ||||||
| ### Compiling | ### Build APT package | ||||||
| Reminder Bot can be built by running `cargo build --release` in the top level directory. It is necessary to create a folder called 'assets' containing an image file with its name specified in the environment as `WEBHOOK_AVATAR`, of dimensions 128x128px to be used as the webhook avatar. |  | ||||||
|  |  | ||||||
| #### Compilation environment variables | Recommended method. | ||||||
| These environment variables must be provided when compiling the bot |  | ||||||
| * `DATABASE_URL` - the URL of your MySQL database (`mysql://user[:password]@domain/database`) |  | ||||||
| * `WEBHOOK_AVATAR` - accepts the name of an image file located in `$CARGO_MANIFEST_DIR/assets/` to be used as the avatar when creating webhooks. **IMPORTANT: image file must be 128x128 or smaller in size** |  | ||||||
|  |  | ||||||
| ### Setting up Python | By default, this builds targeting Ubuntu 20.04. Modify the Containerfile if you wish to target a different platform. These instructions are written using `podman`, but `docker` should work too. | ||||||
| Reminder Bot by default looks for a venv within it's working directory to run Python out of. To set up a venv, install `python3-venv` and run `python3 -m venv venv`. Then, run `source venv/bin/activate` to activate the venv, and do `pip install dateparser` to install the required library |  | ||||||
|  | 1. Install container software: `sudo apt install podman`. | ||||||
|  | 2. Install database server: `sudo apt install mysql-server-8.0`. Create a database called `reminders` | ||||||
|  | 3. Install SQLx CLI: `cargo install sqlx-cli` | ||||||
|  | 4. From the source code directory, execute `sqlx migrate run` | ||||||
|  | 5. Build container image: `podman build -t reminder-rs .` | ||||||
|  | 6. Build with podman: `podman run --rm --network=host -v "$PWD":/mnt -w /mnt -e "DATABASE_URL=mysql://user@localhost/reminders" reminder-rs cargo deb`  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ### Compiling for other target | ||||||
|  |  | ||||||
|  | 1. Install requirements:  | ||||||
|  | `sudo apt install gcc gcc-multilib cmake libssl-dev build-essential python3-dateparser` | ||||||
|  | 2. Install rustup from https://rustup.rs | ||||||
|  | 3. Install the nightly toolchain: `rustup toolchain default nightly` | ||||||
|  | 4. Install database server: `sudo apt install mysql-server-8.0`. Create a database called `reminders`. | ||||||
|  | 5. Install `sqlx-cli`: `cargo install sqlx-cli`. | ||||||
|  | 6. Run migrations: `sqlx migrate run`. | ||||||
|  | 7. Set environment variables: | ||||||
|  |    * `DATABASE_URL` - the URL of your MySQL database (`mysql://user[:password]@domain/database`) | ||||||
|  | 8. Build: `cargo build --release` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ### Configuring | ||||||
|  |  | ||||||
| ### Environment Variables |  | ||||||
| Reminder Bot reads a number of environment variables. Some are essential, and others have hardcoded fallbacks. Environment variables can be loaded from a .env file in the working directory. | Reminder Bot reads a number of environment variables. Some are essential, and others have hardcoded fallbacks. Environment variables can be loaded from a .env file in the working directory. | ||||||
|  |  | ||||||
| __Required Variables__ | __Required Variables__ | ||||||
| @@ -30,10 +48,5 @@ __Other Variables__ | |||||||
| * `LOCAL_TIMEZONE` - default `UTC`, necessary for calculations in the natural language processor | * `LOCAL_TIMEZONE` - default `UTC`, necessary for calculations in the natural language processor | ||||||
| * `SUBSCRIPTION_ROLES` - default `None`, accepts a list of Discord role IDs that are given to subscribed users | * `SUBSCRIPTION_ROLES` - default `None`, accepts a list of Discord role IDs that are given to subscribed users | ||||||
| * `CNC_GUILD` - default `None`, accepts a single Discord guild ID for the server that the subscription roles belong to | * `CNC_GUILD` - default `None`, accepts a single Discord guild ID for the server that the subscription roles belong to | ||||||
| * `PYTHON_LOCATION` - default `venv/bin/python3`. Can be changed if your Python executable is located somewhere else | * `PYTHON_LOCATION` - default `/usr/bin/python3`. Can be changed if your Python executable is located somewhere else | ||||||
| * `THEME_COLOR` - default `8fb677`. Specifies the hex value of the color to use on info message embeds  | * `THEME_COLOR` - default `8fb677`. Specifies the hex value of the color to use on info message embeds  | ||||||
| * `DM_ENABLED` - default `1`, if `1`, Reminder Bot will respond to direct messages |  | ||||||
|  |  | ||||||
| ### Todo List |  | ||||||
|  |  | ||||||
| * Convert aliases to macros |  | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								Rocket.toml
									
									
									
									
									
								
							
							
						
						| @@ -1,6 +1,6 @@ | |||||||
| [default] | [default] | ||||||
| address = "0.0.0.0" | address = "0.0.0.0" | ||||||
| port = 5000 | port = 18920 | ||||||
| template_dir = "web/templates" | template_dir = "web/templates" | ||||||
| limits = { json = "10MiB" } | limits = { json = "10MiB" } | ||||||
|  |  | ||||||
| @@ -11,18 +11,18 @@ secret_key = "tR8krio5FXTnnyIZNiJDXPondz0kI1v6X6BXZcBGIRY=" | |||||||
| certs = "web/private/rsa_sha256_cert.pem" | certs = "web/private/rsa_sha256_cert.pem" | ||||||
| key = "web/private/rsa_sha256_key.pem" | key = "web/private/rsa_sha256_key.pem" | ||||||
|  |  | ||||||
| [rsa_sha256.tls] | [debug.rsa_sha256.tls] | ||||||
| certs = "web/private/rsa_sha256_cert.pem" | certs = "web/private/rsa_sha256_cert.pem" | ||||||
| key = "web/private/rsa_sha256_key.pem" | key = "web/private/rsa_sha256_key.pem" | ||||||
|  |  | ||||||
| [ecdsa_nistp256_sha256.tls] | [debug.ecdsa_nistp256_sha256.tls] | ||||||
| certs = "web/private/ecdsa_nistp256_sha256_cert.pem" | certs = "web/private/ecdsa_nistp256_sha256_cert.pem" | ||||||
| key = "web/private/ecdsa_nistp256_sha256_key_pkcs8.pem" | key = "web/private/ecdsa_nistp256_sha256_key_pkcs8.pem" | ||||||
|  |  | ||||||
| [ecdsa_nistp384_sha384.tls] | [debug.ecdsa_nistp384_sha384.tls] | ||||||
| certs = "web/private/ecdsa_nistp384_sha384_cert.pem" | certs = "web/private/ecdsa_nistp384_sha384_cert.pem" | ||||||
| key = "web/private/ecdsa_nistp384_sha384_key_pkcs8.pem" | key = "web/private/ecdsa_nistp384_sha384_key_pkcs8.pem" | ||||||
|  |  | ||||||
| [ed25519.tls] | [debug.ed25519.tls] | ||||||
| certs = "web/private/ed25519_cert.pem" | certs = "web/private/ed25519_cert.pem" | ||||||
| key = "eb/private/ed25519_key.pem" | key = "eb/private/ed25519_key.pem" | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								assets/webhook.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 21 KiB | 
							
								
								
									
										3
									
								
								build.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | fn main() { | ||||||
|  |     println!("cargo:rerun-if-changed=migrations"); | ||||||
|  | } | ||||||
							
								
								
									
										8
									
								
								conf/Rocket.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | |||||||
|  | [default] | ||||||
|  | address = "127.0.0.1" | ||||||
|  | port = 18920 | ||||||
|  | template_dir = "/lib/reminder-rs/templates" | ||||||
|  | limits = { json = "10MiB" } | ||||||
|  |  | ||||||
|  | [release] | ||||||
|  | # secret_key = "" | ||||||
							
								
								
									
										19
									
								
								conf/default.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | |||||||
|  | DATABASE_URL= | ||||||
|  |  | ||||||
|  | DISCORD_TOKEN= | ||||||
|  | PATREON_GUILD_ID= | ||||||
|  | PATREON_ROLE_ID= | ||||||
|  |  | ||||||
|  | LOCAL_TIMEZONE= | ||||||
|  | MIN_INTERVAL= | ||||||
|  | PYTHON_LOCATION=/usr/bin/python3 | ||||||
|  | DONTRUN= | ||||||
|  | SECRET_KEY= | ||||||
|  |  | ||||||
|  | REMIND_INTERVAL= | ||||||
|  | OAUTH2_DISCORD_CALLBACK= | ||||||
|  | OAUTH2_CLIENT_ID= | ||||||
|  | OAUTH2_CLIENT_SECRET= | ||||||
|  |  | ||||||
|  | REPORT_EMAIL= | ||||||
|  | LOG_TO_DATABASE=1 | ||||||
							
								
								
									
										1
									
								
								cron.d/reminder_health
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | */10 * * * * reminder /lib/reminder-rs/healthcheck | ||||||
							
								
								
									
										9
									
								
								debian/postinst
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | |||||||
|  | #!/bin/bash | ||||||
|  |  | ||||||
|  | set -e | ||||||
|  |  | ||||||
|  | id -u reminder &>/dev/null || useradd -r -M reminder | ||||||
|  |  | ||||||
|  | chown -R reminder /etc/reminder-rs | ||||||
|  |  | ||||||
|  | #DEBHELPER# | ||||||
							
								
								
									
										7
									
								
								debian/postrm
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | |||||||
|  | #!/bin/bash | ||||||
|  |  | ||||||
|  | set -e | ||||||
|  |  | ||||||
|  | id -u reminder &>/dev/null || userdel reminder | ||||||
|  |  | ||||||
|  | #DEBHELPER# | ||||||
							
								
								
									
										13
									
								
								healthcheck
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,13 @@ | |||||||
|  | #!/bin/bash | ||||||
|  |  | ||||||
|  | export $(grep -v '^#' /etc/reminder-rs/config.env | xargs -d '\n') | ||||||
|  |  | ||||||
|  | REGEX='mysql://([A-Za-z]+)@(.+)/(.+)' | ||||||
|  | [[ $DATABASE_URL =~ $REGEX ]] | ||||||
|  |  | ||||||
|  | VAR=$(mysql -u "${BASH_REMATCH[1]}" -h "${BASH_REMATCH[2]}" -N -D "${BASH_REMATCH[3]}" -e "SELECT COUNT(1) FROM reminders WHERE utc_time < NOW() - INTERVAL 10 MINUTE AND enabled = 1 AND status = 'pending'") | ||||||
|  |  | ||||||
|  | if [ "$VAR" -gt 0 ] | ||||||
|  | then | ||||||
|  |   echo "This is to inform that there is a reminder backlog which must be resolved." | mail -s "Backlog: $VAR" "$REPORT_EMAIL" | ||||||
|  | fi | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| USE reminders; |  | ||||||
|  |  | ||||||
| ALTER TABLE reminders.reminders RENAME COLUMN `interval` TO `interval_seconds`; |  | ||||||
| ALTER TABLE reminders.reminders ADD COLUMN `interval_months` INT UNSIGNED DEFAULT NULL; |  | ||||||
| @@ -1,10 +1,6 @@ | |||||||
| CREATE DATABASE IF NOT EXISTS reminders; |  | ||||||
| 
 |  | ||||||
| SET FOREIGN_KEY_CHECKS=0; | SET FOREIGN_KEY_CHECKS=0; | ||||||
| 
 | 
 | ||||||
| USE reminders; | CREATE TABLE guilds ( | ||||||
| 
 |  | ||||||
| CREATE TABLE reminders.guilds ( |  | ||||||
|     id INT UNSIGNED UNIQUE NOT NULL AUTO_INCREMENT, |     id INT UNSIGNED UNIQUE NOT NULL AUTO_INCREMENT, | ||||||
|     guild BIGINT UNSIGNED UNIQUE NOT NULL, |     guild BIGINT UNSIGNED UNIQUE NOT NULL, | ||||||
| 
 | 
 | ||||||
| @@ -18,10 +14,10 @@ CREATE TABLE reminders.guilds ( | |||||||
|     default_avatar VARCHAR(512) DEFAULT 'https://raw.githubusercontent.com/reminder-bot/logos/master/Remind_Me_Bot_Logo_PPic.jpg' NOT NULL, |     default_avatar VARCHAR(512) DEFAULT 'https://raw.githubusercontent.com/reminder-bot/logos/master/Remind_Me_Bot_Logo_PPic.jpg' NOT NULL, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (default_channel_id) REFERENCES reminders.channels(id) ON DELETE SET NULL |     FOREIGN KEY (default_channel_id) REFERENCES channels(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.channels ( | CREATE TABLE channels ( | ||||||
|     id INT UNSIGNED UNIQUE NOT NULL AUTO_INCREMENT, |     id INT UNSIGNED UNIQUE NOT NULL AUTO_INCREMENT, | ||||||
|     channel BIGINT UNSIGNED UNIQUE NOT NULL, |     channel BIGINT UNSIGNED UNIQUE NOT NULL, | ||||||
| 
 | 
 | ||||||
| @@ -39,10 +35,10 @@ CREATE TABLE reminders.channels ( | |||||||
|     guild_id INT UNSIGNED, |     guild_id INT UNSIGNED, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE |     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.users ( | CREATE TABLE users ( | ||||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, |     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||||
|     user BIGINT UNSIGNED UNIQUE NOT NULL, |     user BIGINT UNSIGNED UNIQUE NOT NULL, | ||||||
| 
 | 
 | ||||||
| @@ -59,10 +55,10 @@ CREATE TABLE reminders.users ( | |||||||
|     patreon BOOLEAN NOT NULL DEFAULT 0, |     patreon BOOLEAN NOT NULL DEFAULT 0, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (dm_channel) REFERENCES reminders.channels(id) ON DELETE RESTRICT |     FOREIGN KEY (dm_channel) REFERENCES channels(id) ON DELETE RESTRICT | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.roles ( | CREATE TABLE roles ( | ||||||
|     id INT UNSIGNED UNIQUE NOT NULL AUTO_INCREMENT, |     id INT UNSIGNED UNIQUE NOT NULL AUTO_INCREMENT, | ||||||
|     role BIGINT UNSIGNED UNIQUE NOT NULL, |     role BIGINT UNSIGNED UNIQUE NOT NULL, | ||||||
| 
 | 
 | ||||||
| @@ -71,10 +67,10 @@ CREATE TABLE reminders.roles ( | |||||||
|     guild_id INT UNSIGNED NOT NULL, |     guild_id INT UNSIGNED NOT NULL, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE |     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.embeds ( | CREATE TABLE embeds ( | ||||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, |     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||||
| 
 | 
 | ||||||
|     title VARCHAR(256) NOT NULL DEFAULT '', |     title VARCHAR(256) NOT NULL DEFAULT '', | ||||||
| @@ -91,7 +87,7 @@ CREATE TABLE reminders.embeds ( | |||||||
|     PRIMARY KEY (id) |     PRIMARY KEY (id) | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.embed_fields ( | CREATE TABLE embed_fields ( | ||||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, |     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||||
| 
 | 
 | ||||||
|     title VARCHAR(256) NOT NULL DEFAULT '', |     title VARCHAR(256) NOT NULL DEFAULT '', | ||||||
| @@ -100,10 +96,10 @@ CREATE TABLE reminders.embed_fields ( | |||||||
|     embed_id INT UNSIGNED NOT NULL, |     embed_id INT UNSIGNED NOT NULL, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (embed_id) REFERENCES reminders.embeds(id) ON DELETE CASCADE |     FOREIGN KEY (embed_id) REFERENCES embeds(id) ON DELETE CASCADE | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.messages ( | CREATE TABLE messages ( | ||||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, |     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||||
| 
 | 
 | ||||||
|     content VARCHAR(2048) NOT NULL DEFAULT '', |     content VARCHAR(2048) NOT NULL DEFAULT '', | ||||||
| @@ -114,10 +110,10 @@ CREATE TABLE reminders.messages ( | |||||||
|     attachment_name VARCHAR(260), |     attachment_name VARCHAR(260), | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (embed_id) REFERENCES reminders.embeds(id) ON DELETE SET NULL |     FOREIGN KEY (embed_id) REFERENCES embeds(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.reminders ( | CREATE TABLE reminders ( | ||||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, |     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||||
|     uid VARCHAR(64) UNIQUE NOT NULL, |     uid VARCHAR(64) UNIQUE NOT NULL, | ||||||
| 
 | 
 | ||||||
| @@ -140,20 +136,20 @@ CREATE TABLE reminders.reminders ( | |||||||
|     set_by INT UNSIGNED, |     set_by INT UNSIGNED, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (message_id) REFERENCES reminders.messages(id) ON DELETE RESTRICT, |     FOREIGN KEY (message_id) REFERENCES messages(id) ON DELETE RESTRICT, | ||||||
|     FOREIGN KEY (channel_id) REFERENCES reminders.channels(id) ON DELETE CASCADE, |     FOREIGN KEY (channel_id) REFERENCES channels(id) ON DELETE CASCADE, | ||||||
|     FOREIGN KEY (set_by) REFERENCES reminders.users(id) ON DELETE SET NULL |     FOREIGN KEY (set_by) REFERENCES users(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TRIGGER message_cleanup AFTER DELETE ON reminders.reminders | CREATE TRIGGER message_cleanup AFTER DELETE ON reminders | ||||||
| FOR EACH ROW | FOR EACH ROW | ||||||
|     DELETE FROM reminders.messages WHERE id = OLD.message_id; |     DELETE FROM messages WHERE id = OLD.message_id; | ||||||
| 
 | 
 | ||||||
| CREATE TRIGGER embed_cleanup AFTER DELETE ON reminders.messages | CREATE TRIGGER embed_cleanup AFTER DELETE ON messages | ||||||
| FOR EACH ROW | FOR EACH ROW | ||||||
|     DELETE FROM reminders.embeds WHERE id = OLD.embed_id; |     DELETE FROM embeds WHERE id = OLD.embed_id; | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.todos ( | CREATE TABLE todos ( | ||||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, |     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||||
|     user_id INT UNSIGNED, |     user_id INT UNSIGNED, | ||||||
|     guild_id INT UNSIGNED, |     guild_id INT UNSIGNED, | ||||||
| @@ -161,23 +157,23 @@ CREATE TABLE reminders.todos ( | |||||||
|     value VARCHAR(2000) NOT NULL, |     value VARCHAR(2000) NOT NULL, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (user_id) REFERENCES reminders.users(id) ON DELETE SET NULL, |     FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE SET NULL, | ||||||
|     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE, |     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE, | ||||||
|     FOREIGN KEY (channel_id) REFERENCES reminders.channels(id) ON DELETE SET NULL |     FOREIGN KEY (channel_id) REFERENCES channels(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.command_restrictions ( | CREATE TABLE command_restrictions ( | ||||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, |     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||||
| 
 | 
 | ||||||
|     role_id INT UNSIGNED NOT NULL, |     role_id INT UNSIGNED NOT NULL, | ||||||
|     command ENUM('todos', 'natural', 'remind', 'interval', 'timer', 'del', 'look', 'alias', 'countdown') NOT NULL, |     command ENUM('todos', 'natural', 'remind', 'interval', 'timer', 'del', 'look', 'alias', 'countdown') NOT NULL, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (role_id) REFERENCES reminders.roles(id) ON DELETE CASCADE, |     FOREIGN KEY (role_id) REFERENCES roles(id) ON DELETE CASCADE, | ||||||
|     UNIQUE KEY (`role_id`, `command`) |     UNIQUE KEY (`role_id`, `command`) | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.timers ( | CREATE TABLE timers ( | ||||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, |     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||||
|     start_time TIMESTAMP NOT NULL DEFAULT NOW(), |     start_time TIMESTAMP NOT NULL DEFAULT NOW(), | ||||||
|     name VARCHAR(32) NOT NULL, |     name VARCHAR(32) NOT NULL, | ||||||
| @@ -186,7 +182,7 @@ CREATE TABLE reminders.timers ( | |||||||
|     PRIMARY KEY (id) |     PRIMARY KEY (id) | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.events ( | CREATE TABLE events ( | ||||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, |     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||||
|     `time` TIMESTAMP NOT NULL DEFAULT NOW(), |     `time` TIMESTAMP NOT NULL DEFAULT NOW(), | ||||||
| 
 | 
 | ||||||
| @@ -198,12 +194,12 @@ CREATE TABLE reminders.events ( | |||||||
|     reminder_id INT UNSIGNED, |     reminder_id INT UNSIGNED, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE, |     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE, | ||||||
|     FOREIGN KEY (user_id) REFERENCES reminders.users(id) ON DELETE SET NULL, |     FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE SET NULL, | ||||||
|     FOREIGN KEY (reminder_id) REFERENCES reminders.reminders(id) ON DELETE SET NULL |     FOREIGN KEY (reminder_id) REFERENCES reminders(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.command_aliases ( | CREATE TABLE command_aliases ( | ||||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, |     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||||
| 
 | 
 | ||||||
|     guild_id INT UNSIGNED NOT NULL, |     guild_id INT UNSIGNED NOT NULL, | ||||||
| @@ -212,22 +208,22 @@ CREATE TABLE reminders.command_aliases ( | |||||||
|     command VARCHAR(2048) NOT NULL, |     command VARCHAR(2048) NOT NULL, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE, |     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE, | ||||||
|     UNIQUE KEY (`guild_id`, `name`) |     UNIQUE KEY (`guild_id`, `name`) | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders.guild_users ( | CREATE TABLE guild_users ( | ||||||
|     guild INT UNSIGNED NOT NULL, |     guild INT UNSIGNED NOT NULL, | ||||||
|     user INT UNSIGNED NOT NULL, |     user INT UNSIGNED NOT NULL, | ||||||
| 
 | 
 | ||||||
|     can_access BOOL NOT NULL DEFAULT 0, |     can_access BOOL NOT NULL DEFAULT 0, | ||||||
| 
 | 
 | ||||||
|     FOREIGN KEY (guild) REFERENCES reminders.guilds(id) ON DELETE CASCADE, |     FOREIGN KEY (guild) REFERENCES guilds(id) ON DELETE CASCADE, | ||||||
|     FOREIGN KEY (user) REFERENCES reminders.users(id) ON DELETE CASCADE, |     FOREIGN KEY (user) REFERENCES users(id) ON DELETE CASCADE, | ||||||
|     UNIQUE KEY (guild, user) |     UNIQUE KEY (guild, user) | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE EVENT reminders.event_cleanup | CREATE EVENT event_cleanup | ||||||
| ON SCHEDULE AT CURRENT_TIMESTAMP + INTERVAL 1 DAY | ON SCHEDULE AT CURRENT_TIMESTAMP + INTERVAL 1 DAY | ||||||
| ON COMPLETION PRESERVE | ON COMPLETION PRESERVE | ||||||
| DO DELETE FROM reminders.events WHERE `time` < DATE_SUB(NOW(), INTERVAL 5 DAY); | DO DELETE FROM events WHERE `time` < DATE_SUB(NOW(), INTERVAL 5 DAY); | ||||||
| @@ -1,5 +1,3 @@ | |||||||
| USE reminders; |  | ||||||
| 
 |  | ||||||
| SET FOREIGN_KEY_CHECKS = 0; | SET FOREIGN_KEY_CHECKS = 0; | ||||||
| 
 | 
 | ||||||
| DROP TABLE IF EXISTS reminders_new; | DROP TABLE IF EXISTS reminders_new; | ||||||
| @@ -157,4 +155,9 @@ CREATE TABLE events ( | |||||||
|     FOREIGN KEY (reminder_id) REFERENCES reminders_new(id) ON DELETE SET NULL |     FOREIGN KEY (reminder_id) REFERENCES reminders_new(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
|  | DROP TABLE reminders; | ||||||
|  | DROP TABLE embed_fields; | ||||||
|  | RENAME TABLE reminders_new TO reminders; | ||||||
|  | RENAME TABLE embed_fields_new TO embed_fields; | ||||||
|  | 
 | ||||||
| SET FOREIGN_KEY_CHECKS = 1; | SET FOREIGN_KEY_CHECKS = 1; | ||||||
| @@ -1,5 +1,3 @@ | |||||||
| USE reminders; |  | ||||||
| 
 |  | ||||||
| CREATE TABLE macro ( | CREATE TABLE macro ( | ||||||
|     id INT UNSIGNED AUTO_INCREMENT, |     id INT UNSIGNED AUTO_INCREMENT, | ||||||
|     guild_id INT UNSIGNED NOT NULL, |     guild_id INT UNSIGNED NOT NULL, | ||||||
| @@ -0,0 +1,2 @@ | |||||||
|  | ALTER TABLE reminders RENAME COLUMN `interval` TO `interval_seconds`; | ||||||
|  | ALTER TABLE reminders ADD COLUMN `interval_months` INT UNSIGNED DEFAULT NULL; | ||||||
| @@ -1,5 +1,3 @@ | |||||||
| USE reminders; |  | ||||||
| 
 |  | ||||||
| CREATE TABLE reminder_template ( | CREATE TABLE reminder_template ( | ||||||
|     `id` INT UNSIGNED NOT NULL AUTO_INCREMENT, |     `id` INT UNSIGNED NOT NULL AUTO_INCREMENT, | ||||||
| 
 | 
 | ||||||
| @@ -32,3 +30,20 @@ CREATE TABLE reminder_template ( | |||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| ALTER TABLE reminders ADD COLUMN embed_fields JSON; | ALTER TABLE reminders ADD COLUMN embed_fields JSON; | ||||||
|  | 
 | ||||||
|  | update reminders | ||||||
|  |     inner join embed_fields as E | ||||||
|  |     on E.reminder_id = reminders.id | ||||||
|  | set embed_fields = ( | ||||||
|  |     select JSON_ARRAYAGG( | ||||||
|  |         JSON_OBJECT( | ||||||
|  |             'title', E.title, | ||||||
|  |             'value', E.value, | ||||||
|  |             'inline', | ||||||
|  |             if(inline = 1, cast(TRUE as json), cast(FALSE as json)) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |     from embed_fields | ||||||
|  |     group by reminder_id | ||||||
|  |     having reminder_id = reminders.id | ||||||
|  |     ); | ||||||
							
								
								
									
										1
									
								
								migrations/20221210000000_reminder_daily_intervals.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | ALTER TABLE reminders ADD COLUMN `interval_days` INT UNSIGNED DEFAULT NULL; | ||||||
							
								
								
									
										1
									
								
								migrations/20230511125236_reminder_threads.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | ALTER TABLE reminders ADD COLUMN `thread_id` BIGINT DEFAULT NULL; | ||||||
							
								
								
									
										1
									
								
								migrations/20230511180231_ephemeral_confirmations.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | ALTER TABLE guilds ADD COLUMN ephemeral_confirmations BOOL NOT NULL DEFAULT 0; | ||||||
							
								
								
									
										2
									
								
								migrations/20230722130906_increase_reminder_name.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,2 @@ | |||||||
|  | ALTER TABLE reminders MODIFY `name` VARCHAR(100) NOT NULL DEFAULT 'Reminder'; | ||||||
|  | ALTER TABLE reminder_template MODIFY `name` VARCHAR(100) NOT NULL DEFAULT 'Reminder'; | ||||||
							
								
								
									
										9
									
								
								migrations/20230730134827_stats.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | |||||||
|  | CREATE TABLE stat ( | ||||||
|  |     `id` BIGINT NOT NULL AUTO_INCREMENT, | ||||||
|  |     `utc_time` DATETIME NOT NULL DEFAULT NOW(), | ||||||
|  |     `type` ENUM('reminder_sent', 'reminder_failed'), | ||||||
|  |     `reminder_id` INT UNSIGNED, | ||||||
|  |     `message` TEXT, | ||||||
|  |  | ||||||
|  |     PRIMARY KEY (`id`) | ||||||
|  | ); | ||||||
							
								
								
									
										2
									
								
								migrations/20230731170452_reminder_archive.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,2 @@ | |||||||
|  | ALTER TABLE reminders ADD COLUMN `status` ENUM ('pending', 'sent', 'failed', 'deleted') NOT NULL DEFAULT 'pending'; | ||||||
|  | ALTER TABLE reminders ADD COLUMN `status_message` TEXT; | ||||||
| @@ -0,0 +1,3 @@ | |||||||
|  | ALTER TABLE `reminder_template` ADD COLUMN `interval_seconds` INT UNSIGNED; | ||||||
|  | ALTER TABLE `reminder_template` ADD COLUMN `interval_days` INT UNSIGNED; | ||||||
|  | ALTER TABLE `reminder_template` ADD COLUMN `interval_months` INT UNSIGNED; | ||||||
							
								
								
									
										41
									
								
								nginx/reminder-rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,41 @@ | |||||||
|  | server { | ||||||
|  |         server_name www.reminder-bot.com; | ||||||
|  |  | ||||||
|  |         return 301 $scheme://reminder-bot.com$request_uri; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | server { | ||||||
|  |         listen 80; | ||||||
|  |         server_name reminder-bot.com; | ||||||
|  |  | ||||||
|  | 	    return 301 https://reminder-bot.com$request_uri; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | server { | ||||||
|  |         listen 443 ssl; | ||||||
|  |         server_name reminder-bot.com; | ||||||
|  |  | ||||||
|  |         ssl_certificate /etc/letsencrypt/live/reminder-bot.com/fullchain.pem; | ||||||
|  |         ssl_certificate_key /etc/letsencrypt/live/reminder-bot.com/privkey.pem; | ||||||
|  |  | ||||||
|  |         access_log /var/log/nginx/access.log; | ||||||
|  |         error_log /var/log/nginx/error.log; | ||||||
|  |  | ||||||
|  |         proxy_buffer_size 128k; | ||||||
|  |         proxy_buffers 4 256k; | ||||||
|  |         proxy_busy_buffers_size 256k; | ||||||
|  |  | ||||||
|  |         location / { | ||||||
|  |                 proxy_pass http://localhost:18920; | ||||||
|  |                 proxy_redirect off; | ||||||
|  |                 proxy_set_header Host $host; | ||||||
|  |                 proxy_set_header X-Real-IP $remote_addr; | ||||||
|  |                 proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; | ||||||
|  | 		        proxy_set_header X-Forwarded-Proto $scheme; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         location /static { | ||||||
|  |                 alias /var/www/reminder-rs/static; | ||||||
|  |                 expires 30d; | ||||||
|  |         } | ||||||
|  | } | ||||||
| @@ -5,14 +5,12 @@ edition = "2021" | |||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| tokio = { version = "1", features = ["process", "full"] } | tokio = { version = "1", features = ["process", "full"] } | ||||||
| regex = "1.4" | regex = "1.9" | ||||||
| log = "0.4" | log = "0.4" | ||||||
| env_logger = "0.8" |  | ||||||
| chrono = "0.4" | chrono = "0.4" | ||||||
| chrono-tz = { version = "0.5", features = ["serde"] } | chrono-tz = { version = "0.8", features = ["serde"] } | ||||||
| lazy_static = "1.4" | lazy_static = "1.4" | ||||||
| num-integer = "0.1" | num-integer = "0.1" | ||||||
| serde = "1.0" | serde = "1.0" | ||||||
| serde_json = "1.0" | sqlx = { version = "0.7", features = ["runtime-tokio-rustls", "macros", "mysql", "bigdecimal", "chrono", "json"]} | ||||||
| sqlx = { version = "0.5", features = ["runtime-tokio-rustls", "macros", "mysql", "bigdecimal", "chrono", "json"]} | serenity = { version = "0.11", default-features = false, features = ["builder", "cache", "client", "gateway", "http", "model", "utils", "rustls_backend"] } | ||||||
| serenity = { version = "0.11.1", default-features = false, features = ["builder", "cache", "client", "gateway", "http", "model", "utils", "rustls_backend"] } |  | ||||||
|   | |||||||
| @@ -1,4 +1,6 @@ | |||||||
| use chrono::Duration; | use std::env; | ||||||
|  |  | ||||||
|  | use chrono::{DateTime, Days, Duration, Months}; | ||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
| use lazy_static::lazy_static; | use lazy_static::lazy_static; | ||||||
| use log::{error, info, warn}; | use log::{error, info, warn}; | ||||||
| @@ -7,7 +9,7 @@ use regex::{Captures, Regex}; | |||||||
| use serde::Deserialize; | use serde::Deserialize; | ||||||
| use serenity::{ | use serenity::{ | ||||||
|     builder::CreateEmbed, |     builder::CreateEmbed, | ||||||
|     http::{CacheHttp, Http, StatusCode}, |     http::{CacheHttp, Http, HttpError}, | ||||||
|     model::{ |     model::{ | ||||||
|         channel::{Channel, Embed as SerenityEmbed}, |         channel::{Channel, Embed as SerenityEmbed}, | ||||||
|         id::ChannelId, |         id::ChannelId, | ||||||
| @@ -30,6 +32,7 @@ lazy_static! { | |||||||
|         Regex::new(r#"<<timefrom:(?P<time>\d+):(?P<format>.+)?>>"#).unwrap(); |         Regex::new(r#"<<timefrom:(?P<time>\d+):(?P<format>.+)?>>"#).unwrap(); | ||||||
|     pub static ref TIMENOW_REGEX: Regex = |     pub static ref TIMENOW_REGEX: Regex = | ||||||
|         Regex::new(r#"<<timenow:(?P<timezone>(?:\w|/|_)+):(?P<format>.+)?>>"#).unwrap(); |         Regex::new(r#"<<timenow:(?P<timezone>(?:\w|/|_)+):(?P<format>.+)?>>"#).unwrap(); | ||||||
|  |     pub static ref LOG_TO_DATABASE: bool = env::var("LOG_TO_DATABASE").map_or(true, |v| v == "1"); | ||||||
| } | } | ||||||
|  |  | ||||||
| fn fmt_displacement(format: &str, seconds: u64) -> String { | fn fmt_displacement(format: &str, seconds: u64) -> String { | ||||||
| @@ -58,22 +61,27 @@ fn fmt_displacement(format: &str, seconds: u64) -> String { | |||||||
|  |  | ||||||
| pub fn substitute(string: &str) -> String { | pub fn substitute(string: &str) -> String { | ||||||
|     let new = TIMEFROM_REGEX.replace(string, |caps: &Captures| { |     let new = TIMEFROM_REGEX.replace(string, |caps: &Captures| { | ||||||
|         let final_time = caps.name("time").unwrap().as_str(); |         let final_time = caps.name("time").map(|m| m.as_str().parse::<i64>().ok()).flatten(); | ||||||
|         let format = caps.name("format").unwrap().as_str(); |         let format = caps.name("format").map(|m| m.as_str()); | ||||||
|  |  | ||||||
|         if let Ok(final_time) = final_time.parse::<i64>() { |         if let (Some(final_time), Some(format)) = (final_time, format) { | ||||||
|             let dt = NaiveDateTime::from_timestamp(final_time, 0); |             match NaiveDateTime::from_timestamp_opt(final_time, 0) { | ||||||
|             let now = Utc::now().naive_utc(); |                 Some(dt) => { | ||||||
|  |                     let now = Utc::now().naive_utc(); | ||||||
|  |  | ||||||
|             let difference = { |                     let difference = { | ||||||
|                 if now < dt { |                         if now < dt { | ||||||
|                     dt - Utc::now().naive_utc() |                             dt - Utc::now().naive_utc() | ||||||
|                 } else { |                         } else { | ||||||
|                     Utc::now().naive_utc() - dt |                             Utc::now().naive_utc() - dt | ||||||
|  |                         } | ||||||
|  |                     }; | ||||||
|  |  | ||||||
|  |                     fmt_displacement(format, difference.num_seconds() as u64) | ||||||
|                 } |                 } | ||||||
|             }; |  | ||||||
|  |  | ||||||
|             fmt_displacement(format, difference.num_seconds() as u64) |                 None => String::new(), | ||||||
|  |             } | ||||||
|         } else { |         } else { | ||||||
|             String::new() |             String::new() | ||||||
|         } |         } | ||||||
| @@ -81,13 +89,11 @@ pub fn substitute(string: &str) -> String { | |||||||
|  |  | ||||||
|     TIMENOW_REGEX |     TIMENOW_REGEX | ||||||
|         .replace(&new, |caps: &Captures| { |         .replace(&new, |caps: &Captures| { | ||||||
|             let timezone = caps.name("timezone").unwrap().as_str(); |             let timezone = caps.name("timezone").map(|m| m.as_str().parse::<Tz>().ok()).flatten(); | ||||||
|  |             let format = caps.name("format").map(|m| m.as_str()); | ||||||
|  |  | ||||||
|             println!("{}", timezone); |             if let (Some(timezone), Some(format)) = (timezone, format) { | ||||||
|  |                 let now = Utc::now().with_timezone(&timezone); | ||||||
|             if let Ok(tz) = timezone.parse::<Tz>() { |  | ||||||
|                 let format = caps.name("format").unwrap().as_str(); |  | ||||||
|                 let now = Utc::now().with_timezone(&tz); |  | ||||||
|  |  | ||||||
|                 now.format(format).to_string() |                 now.format(format).to_string() | ||||||
|             } else { |             } else { | ||||||
| @@ -122,7 +128,7 @@ impl Embed { | |||||||
|         pool: impl Executor<'_, Database = Database> + Copy, |         pool: impl Executor<'_, Database = Database> + Copy, | ||||||
|         id: u32, |         id: u32, | ||||||
|     ) -> Option<Self> { |     ) -> Option<Self> { | ||||||
|         let mut embed = sqlx::query_as!( |         match sqlx::query_as!( | ||||||
|             Self, |             Self, | ||||||
|             r#" |             r#" | ||||||
|             SELECT |             SELECT | ||||||
| @@ -142,21 +148,29 @@ impl Embed { | |||||||
|         ) |         ) | ||||||
|         .fetch_one(pool) |         .fetch_one(pool) | ||||||
|         .await |         .await | ||||||
|         .unwrap(); |         { | ||||||
|  |             Ok(mut embed) => { | ||||||
|  |                 embed.title = substitute(&embed.title); | ||||||
|  |                 embed.description = substitute(&embed.description); | ||||||
|  |                 embed.footer = substitute(&embed.footer); | ||||||
|  |  | ||||||
|         embed.title = substitute(&embed.title); |                 embed.fields.iter_mut().for_each(|field| { | ||||||
|         embed.description = substitute(&embed.description); |                     field.title = substitute(&field.title); | ||||||
|         embed.footer = substitute(&embed.footer); |                     field.value = substitute(&field.value); | ||||||
|  |                 }); | ||||||
|  |  | ||||||
|         embed.fields.iter_mut().for_each(|mut field| { |                 if embed.has_content() { | ||||||
|             field.title = substitute(&field.title); |                     Some(embed) | ||||||
|             field.value = substitute(&field.value); |                 } else { | ||||||
|         }); |                     None | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|         if embed.has_content() { |             Err(e) => { | ||||||
|             Some(embed) |                 warn!("Error loading embed from reminder: {:?}", e); | ||||||
|         } else { |  | ||||||
|             None |                 None | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -220,7 +234,6 @@ impl Into<CreateEmbed> for Embed { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Debug)] |  | ||||||
| pub struct Reminder { | pub struct Reminder { | ||||||
|     id: u32, |     id: u32, | ||||||
|  |  | ||||||
| @@ -238,11 +251,12 @@ pub struct Reminder { | |||||||
|     attachment: Option<Vec<u8>>, |     attachment: Option<Vec<u8>>, | ||||||
|     attachment_name: Option<String>, |     attachment_name: Option<String>, | ||||||
|  |  | ||||||
|     utc_time: NaiveDateTime, |     utc_time: DateTime<Utc>, | ||||||
|     timezone: String, |     timezone: String, | ||||||
|     restartable: bool, |     restartable: bool, | ||||||
|     expires: Option<NaiveDateTime>, |     expires: Option<DateTime<Utc>>, | ||||||
|     interval_seconds: Option<u32>, |     interval_seconds: Option<u32>, | ||||||
|  |     interval_days: Option<u32>, | ||||||
|     interval_months: Option<u32>, |     interval_months: Option<u32>, | ||||||
|  |  | ||||||
|     avatar: Option<String>, |     avatar: Option<String>, | ||||||
| @@ -251,9 +265,9 @@ pub struct Reminder { | |||||||
|  |  | ||||||
| impl Reminder { | impl Reminder { | ||||||
|     pub async fn fetch_reminders(pool: impl Executor<'_, Database = Database> + Copy) -> Vec<Self> { |     pub async fn fetch_reminders(pool: impl Executor<'_, Database = Database> + Copy) -> Vec<Self> { | ||||||
|         sqlx::query_as_unchecked!( |         match sqlx::query_as_unchecked!( | ||||||
|             Reminder, |             Reminder, | ||||||
|             " |             r#" | ||||||
| SELECT | SELECT | ||||||
|     reminders.`id` AS id, |     reminders.`id` AS id, | ||||||
|  |  | ||||||
| @@ -261,9 +275,9 @@ SELECT | |||||||
|     channels.`webhook_id` AS webhook_id, |     channels.`webhook_id` AS webhook_id, | ||||||
|     channels.`webhook_token` AS webhook_token, |     channels.`webhook_token` AS webhook_token, | ||||||
|  |  | ||||||
|     channels.`paused` AS channel_paused, |     channels.`paused` AS 'channel_paused', | ||||||
|     channels.`paused_until` AS channel_paused_until, |     channels.`paused_until` AS 'channel_paused_until', | ||||||
|     reminders.`enabled` AS enabled, |     reminders.`enabled` AS 'enabled', | ||||||
|  |  | ||||||
|     reminders.`tts` AS tts, |     reminders.`tts` AS tts, | ||||||
|     reminders.`pin` AS pin, |     reminders.`pin` AS pin, | ||||||
| @@ -274,8 +288,9 @@ SELECT | |||||||
|     reminders.`utc_time` AS 'utc_time', |     reminders.`utc_time` AS 'utc_time', | ||||||
|     reminders.`timezone` AS timezone, |     reminders.`timezone` AS timezone, | ||||||
|     reminders.`restartable` AS restartable, |     reminders.`restartable` AS restartable, | ||||||
|     reminders.`expires` AS expires, |     reminders.`expires` AS 'expires', | ||||||
|     reminders.`interval_seconds` AS 'interval_seconds', |     reminders.`interval_seconds` AS 'interval_seconds', | ||||||
|  |     reminders.`interval_days` AS 'interval_days', | ||||||
|     reminders.`interval_months` AS 'interval_months', |     reminders.`interval_months` AS 'interval_months', | ||||||
|  |  | ||||||
|     reminders.`avatar` AS avatar, |     reminders.`avatar` AS avatar, | ||||||
| @@ -287,26 +302,48 @@ INNER JOIN | |||||||
| ON | ON | ||||||
|     reminders.channel_id = channels.id |     reminders.channel_id = channels.id | ||||||
| WHERE | WHERE | ||||||
|     reminders.`utc_time` < NOW() |     reminders.`status` = 'pending' AND | ||||||
|             ", |     reminders.`id` IN ( | ||||||
|  |         SELECT | ||||||
|  |             MIN(id) | ||||||
|  |         FROM | ||||||
|  |             reminders | ||||||
|  |         WHERE | ||||||
|  |             reminders.`utc_time` <= NOW() AND | ||||||
|  |             `status` = 'pending' AND | ||||||
|  |             ( | ||||||
|  |                 reminders.`interval_seconds` IS NOT NULL | ||||||
|  |                 OR reminders.`interval_months` IS NOT NULL | ||||||
|  |                 OR reminders.`interval_days` IS NOT NULL | ||||||
|  |                 OR reminders.enabled | ||||||
|  |             ) | ||||||
|  |         GROUP BY channel_id | ||||||
|  |     ) | ||||||
|  |     "#, | ||||||
|         ) |         ) | ||||||
|         .fetch_all(pool) |         .fetch_all(pool) | ||||||
|         .await |         .await | ||||||
|         .unwrap() |         { | ||||||
|         .into_iter() |             Ok(reminders) => reminders | ||||||
|         .map(|mut rem| { |                 .into_iter() | ||||||
|             rem.content = substitute(&rem.content); |                 .map(|mut rem| { | ||||||
|  |                     rem.content = substitute(&rem.content); | ||||||
|  |  | ||||||
|             rem |                     rem | ||||||
|         }) |                 }) | ||||||
|         .collect::<Vec<Self>>() |                 .collect::<Vec<Self>>(), | ||||||
|  |  | ||||||
|  |             Err(e) => { | ||||||
|  |                 warn!("Could not fetch reminders: {:?}", e); | ||||||
|  |  | ||||||
|  |                 vec![] | ||||||
|  |             } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     async fn reset_webhook(&self, pool: impl Executor<'_, Database = Database> + Copy) { |     async fn reset_webhook(&self, pool: impl Executor<'_, Database = Database> + Copy) { | ||||||
|         let _ = sqlx::query!( |         let _ = sqlx::query!( | ||||||
|             " |             "UPDATE channels SET webhook_id = NULL, webhook_token = NULL WHERE channel = ?", | ||||||
| UPDATE channels SET webhook_id = NULL, webhook_token = NULL WHERE channel = ? |  | ||||||
|             ", |  | ||||||
|             self.channel_id |             self.channel_id | ||||||
|         ) |         ) | ||||||
|         .execute(pool) |         .execute(pool) | ||||||
| @@ -314,40 +351,72 @@ UPDATE channels SET webhook_id = NULL, webhook_token = NULL WHERE channel = ? | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     async fn refresh(&self, pool: impl Executor<'_, Database = Database> + Copy) { |     async fn refresh(&self, pool: impl Executor<'_, Database = Database> + Copy) { | ||||||
|         if self.interval_seconds.is_some() || self.interval_months.is_some() { |         if self.interval_seconds.is_some() | ||||||
|             let now = Utc::now().naive_local(); |             || self.interval_months.is_some() | ||||||
|             let mut updated_reminder_time = self.utc_time; |             || self.interval_days.is_some() | ||||||
|  |         { | ||||||
|             if let Some(interval) = self.interval_months { |             // If all intervals are zero then dont care | ||||||
|                 let row = sqlx::query!( |             if self.interval_seconds == Some(0) | ||||||
|                     // use the second date_add to force return value to datetime |                 && self.interval_days == Some(0) | ||||||
|                     "SELECT DATE_ADD(DATE_ADD(?, INTERVAL ? MONTH), INTERVAL 0 SECOND) AS new_time", |                 && self.interval_months == Some(0) | ||||||
|                     updated_reminder_time, |             { | ||||||
|                     interval |                 self.set_sent(pool).await; | ||||||
|                 ) |  | ||||||
|                 .fetch_one(pool) |  | ||||||
|                 .await |  | ||||||
|                 .unwrap(); |  | ||||||
|  |  | ||||||
|                 updated_reminder_time = row.new_time.unwrap(); |  | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             if let Some(interval) = self.interval_seconds { |             let now = Utc::now(); | ||||||
|                 while updated_reminder_time < now { |             let mut updated_reminder_time = | ||||||
|  |                 self.utc_time.with_timezone(&self.timezone.parse().unwrap_or(Tz::UTC)); | ||||||
|  |             let mut fail_count = 0; | ||||||
|  |  | ||||||
|  |             while updated_reminder_time < now && fail_count < 4 { | ||||||
|  |                 if let Some(interval) = self.interval_months { | ||||||
|  |                     if interval != 0 { | ||||||
|  |                         updated_reminder_time = updated_reminder_time | ||||||
|  |                             .checked_add_months(Months::new(interval)) | ||||||
|  |                             .unwrap_or_else(|| { | ||||||
|  |                                 warn!( | ||||||
|  |                                     "{}: Could not add {} months to a reminder", | ||||||
|  |                                     interval, self.id | ||||||
|  |                                 ); | ||||||
|  |                                 fail_count += 1; | ||||||
|  |  | ||||||
|  |                                 updated_reminder_time | ||||||
|  |                             }); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 if let Some(interval) = self.interval_days { | ||||||
|  |                     if interval != 0 { | ||||||
|  |                         updated_reminder_time = updated_reminder_time | ||||||
|  |                             .checked_add_days(Days::new(interval as u64)) | ||||||
|  |                             .unwrap_or_else(|| { | ||||||
|  |                                 warn!("{}: Could not add {} days to a reminder", self.id, interval); | ||||||
|  |                                 fail_count += 1; | ||||||
|  |  | ||||||
|  |                                 updated_reminder_time | ||||||
|  |                             }) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 if let Some(interval) = self.interval_seconds { | ||||||
|                     updated_reminder_time += Duration::seconds(interval as i64); |                     updated_reminder_time += Duration::seconds(interval as i64); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             if self.expires.map_or(false, |expires| { |             if fail_count >= 4 { | ||||||
|                 NaiveDateTime::from_timestamp(updated_reminder_time.timestamp(), 0) > expires |                 self.log_error( | ||||||
|             }) { |                     pool, | ||||||
|                 self.force_delete(pool).await; |                     "Failed to update 4 times and so is being deleted", | ||||||
|  |                     None::<&'static str>, | ||||||
|  |                 ) | ||||||
|  |                 .await; | ||||||
|  |                 self.set_failed(pool, "Failed to update 4 times and so is being deleted").await; | ||||||
|  |             } else if self.expires.map_or(false, |expires| updated_reminder_time > expires) { | ||||||
|  |                 self.set_sent(pool).await; | ||||||
|             } else { |             } else { | ||||||
|                 sqlx::query!( |                 sqlx::query!( | ||||||
|                     " |                     "UPDATE reminders SET `utc_time` = ? WHERE `id` = ?", | ||||||
| UPDATE reminders SET `utc_time` = ? WHERE `id` = ? |                     updated_reminder_time.with_timezone(&Utc), | ||||||
|                     ", |  | ||||||
|                     updated_reminder_time, |  | ||||||
|                     self.id |                     self.id | ||||||
|                 ) |                 ) | ||||||
|                 .execute(pool) |                 .execute(pool) | ||||||
| @@ -355,15 +424,67 @@ UPDATE reminders SET `utc_time` = ? WHERE `id` = ? | |||||||
|                 .expect(&format!("Could not update time on Reminder {}", self.id)); |                 .expect(&format!("Could not update time on Reminder {}", self.id)); | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             self.force_delete(pool).await; |             self.set_sent(pool).await; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     async fn force_delete(&self, pool: impl Executor<'_, Database = Database> + Copy) { |     async fn log_error( | ||||||
|  |         &self, | ||||||
|  |         pool: impl Executor<'_, Database = Database> + Copy, | ||||||
|  |         error: &'static str, | ||||||
|  |         debug_info: Option<impl std::fmt::Debug>, | ||||||
|  |     ) { | ||||||
|  |         let message = match debug_info { | ||||||
|  |             Some(info) => format!( | ||||||
|  |                 "{} | ||||||
|  | {:?}", | ||||||
|  |                 error, info | ||||||
|  |             ), | ||||||
|  |  | ||||||
|  |             None => error.to_string(), | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         error!("[Reminder {}] {}", self.id, message); | ||||||
|  |  | ||||||
|  |         if *LOG_TO_DATABASE { | ||||||
|  |             sqlx::query!( | ||||||
|  |                 "INSERT INTO stat (type, reminder_id, message) VALUES ('reminder_failed', ?, ?)", | ||||||
|  |                 self.id, | ||||||
|  |                 message, | ||||||
|  |             ) | ||||||
|  |             .execute(pool) | ||||||
|  |             .await | ||||||
|  |             .expect("Could not log error to database"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async fn log_success(&self, pool: impl Executor<'_, Database = Database> + Copy) { | ||||||
|  |         if *LOG_TO_DATABASE { | ||||||
|  |             sqlx::query!( | ||||||
|  |                 "INSERT INTO stat (type, reminder_id) VALUES ('reminder_sent', ?)", | ||||||
|  |                 self.id, | ||||||
|  |             ) | ||||||
|  |             .execute(pool) | ||||||
|  |             .await | ||||||
|  |             .expect("Could not log success to database"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async fn set_sent(&self, pool: impl Executor<'_, Database = Database> + Copy) { | ||||||
|  |         sqlx::query!("UPDATE reminders SET `status` = 'sent' WHERE `id` = ?", self.id) | ||||||
|  |             .execute(pool) | ||||||
|  |             .await | ||||||
|  |             .expect(&format!("Could not delete Reminder {}", self.id)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async fn set_failed( | ||||||
|  |         &self, | ||||||
|  |         pool: impl Executor<'_, Database = Database> + Copy, | ||||||
|  |         message: &'static str, | ||||||
|  |     ) { | ||||||
|         sqlx::query!( |         sqlx::query!( | ||||||
|             " |             "UPDATE reminders SET `status` = 'failed', `status_message` = ? WHERE `id` = ?", | ||||||
| DELETE FROM reminders WHERE `id` = ? |             message, | ||||||
|             ", |  | ||||||
|             self.id |             self.id | ||||||
|         ) |         ) | ||||||
|         .execute(pool) |         .execute(pool) | ||||||
| @@ -462,7 +583,9 @@ DELETE FROM reminders WHERE `id` = ? | |||||||
|                     w.content(&reminder.content).tts(reminder.tts); |                     w.content(&reminder.content).tts(reminder.tts); | ||||||
|  |  | ||||||
|                     if let Some(username) = &reminder.username { |                     if let Some(username) = &reminder.username { | ||||||
|                         w.username(username); |                         if !username.is_empty() { | ||||||
|  |                             w.username(username); | ||||||
|  |                         } | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|                     if let Some(avatar) = &reminder.avatar { |                     if let Some(avatar) = &reminder.avatar { | ||||||
| @@ -506,9 +629,7 @@ DELETE FROM reminders WHERE `id` = ? | |||||||
|                     .map_or(true, |inner| inner >= Utc::now().naive_local())) |                     .map_or(true, |inner| inner >= Utc::now().naive_local())) | ||||||
|         { |         { | ||||||
|             let _ = sqlx::query!( |             let _ = sqlx::query!( | ||||||
|                 " |                 "UPDATE `channels` SET paused = 0, paused_until = NULL WHERE `channel` = ?", | ||||||
| UPDATE `channels` SET paused = 0, paused_until = NULL WHERE `channel` = ? |  | ||||||
|                 ", |  | ||||||
|                 self.channel_id |                 self.channel_id | ||||||
|             ) |             ) | ||||||
|             .execute(pool) |             .execute(pool) | ||||||
| @@ -525,7 +646,7 @@ UPDATE `channels` SET paused = 0, paused_until = NULL WHERE `channel` = ? | |||||||
|                 if let Ok(webhook) = webhook_res { |                 if let Ok(webhook) = webhook_res { | ||||||
|                     send_to_webhook(cache_http, &self, webhook, embed).await |                     send_to_webhook(cache_http, &self, webhook, embed).await | ||||||
|                 } else { |                 } else { | ||||||
|                     warn!("Webhook vanished: {:?}", webhook_res); |                     warn!("Webhook vanished for reminder {}: {:?}", self.id, webhook_res); | ||||||
|  |  | ||||||
|                     self.reset_webhook(pool).await; |                     self.reset_webhook(pool).await; | ||||||
|                     send_to_channel(cache_http, &self, embed).await |                     send_to_channel(cache_http, &self, embed).await | ||||||
| @@ -535,19 +656,84 @@ UPDATE `channels` SET paused = 0, paused_until = NULL WHERE `channel` = ? | |||||||
|             }; |             }; | ||||||
|  |  | ||||||
|             if let Err(e) = result { |             if let Err(e) = result { | ||||||
|                 error!("Error sending {:?}: {:?}", self, e); |  | ||||||
|  |  | ||||||
|                 if let Error::Http(error) = e { |                 if let Error::Http(error) = e { | ||||||
|                     if error.status_code() == Some(StatusCode::from_u16(404).unwrap()) { |                     if let HttpError::UnsuccessfulRequest(http_error) = *error { | ||||||
|                         error!("Seeing channel is deleted. Removing reminder"); |                         match http_error.error.code { | ||||||
|                         self.force_delete(pool).await; |                             10003 => { | ||||||
|  |                                 self.log_error( | ||||||
|  |                                     pool, | ||||||
|  |                                     "Could not be sent as channel does not exist", | ||||||
|  |                                     None::<&'static str>, | ||||||
|  |                                 ) | ||||||
|  |                                 .await; | ||||||
|  |                                 self.set_failed( | ||||||
|  |                                     pool, | ||||||
|  |                                     "Could not be sent as channel does not exist", | ||||||
|  |                                 ) | ||||||
|  |                                 .await; | ||||||
|  |                             } | ||||||
|  |                             10004 => { | ||||||
|  |                                 self.log_error( | ||||||
|  |                                     pool, | ||||||
|  |                                     "Could not be sent as guild does not exist", | ||||||
|  |                                     None::<&'static str>, | ||||||
|  |                                 ) | ||||||
|  |                                 .await; | ||||||
|  |                                 self.set_failed(pool, "Could not be sent as guild does not exist") | ||||||
|  |                                     .await; | ||||||
|  |                             } | ||||||
|  |                             50001 => { | ||||||
|  |                                 self.log_error( | ||||||
|  |                                     pool, | ||||||
|  |                                     "Could not be sent as missing access", | ||||||
|  |                                     None::<&'static str>, | ||||||
|  |                                 ) | ||||||
|  |                                 .await; | ||||||
|  |                                 self.set_failed(pool, "Could not be sent as missing access").await; | ||||||
|  |                             } | ||||||
|  |                             50007 => { | ||||||
|  |                                 self.log_error( | ||||||
|  |                                     pool, | ||||||
|  |                                     "Could not be sent as user has DMs disabled", | ||||||
|  |                                     None::<&'static str>, | ||||||
|  |                                 ) | ||||||
|  |                                 .await; | ||||||
|  |                                 self.set_failed(pool, "Could not be sent as user has DMs disabled") | ||||||
|  |                                     .await; | ||||||
|  |                             } | ||||||
|  |                             50013 => { | ||||||
|  |                                 self.log_error( | ||||||
|  |                                     pool, | ||||||
|  |                                     "Could not be sent as permissions are invalid", | ||||||
|  |                                     None::<&'static str>, | ||||||
|  |                                 ) | ||||||
|  |                                 .await; | ||||||
|  |                                 self.set_failed( | ||||||
|  |                                     pool, | ||||||
|  |                                     "Could not be sent as permissions are invalid", | ||||||
|  |                                 ) | ||||||
|  |                                 .await; | ||||||
|  |                             } | ||||||
|  |                             _ => { | ||||||
|  |                                 self.log_error( | ||||||
|  |                                     pool, | ||||||
|  |                                     "HTTP error sending reminder", | ||||||
|  |                                     Some(http_error), | ||||||
|  |                                 ) | ||||||
|  |                                 .await; | ||||||
|  |                                 self.refresh(pool).await; | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|                     } else { |                     } else { | ||||||
|  |                         self.log_error(pool, "(Likely) a parsing error", Some(error)).await; | ||||||
|                         self.refresh(pool).await; |                         self.refresh(pool).await; | ||||||
|                     } |                     } | ||||||
|                 } else { |                 } else { | ||||||
|  |                     self.log_error(pool, "Non-HTTP error", Some(e)).await; | ||||||
|                     self.refresh(pool).await; |                     self.refresh(pool).await; | ||||||
|                 } |                 } | ||||||
|             } else { |             } else { | ||||||
|  |                 self.log_success(pool).await; | ||||||
|                 self.refresh(pool).await; |                 self.refresh(pool).await; | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								reminder-dashboard
									
									
									
									
									
										Submodule
									
								
							
							
								
								
								
								
								
							
						
						
							
								
								
									
										117
									
								
								src/commands/autocomplete.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,117 @@ | |||||||
|  | use std::time::{SystemTime, UNIX_EPOCH}; | ||||||
|  |  | ||||||
|  | use chrono_tz::TZ_VARIANTS; | ||||||
|  | use poise::AutocompleteChoice; | ||||||
|  |  | ||||||
|  | use crate::{models::CtxData, time_parser::natural_parser, Context}; | ||||||
|  |  | ||||||
|  | pub async fn timezone_autocomplete(ctx: Context<'_>, partial: &str) -> Vec<String> { | ||||||
|  |     if partial.is_empty() { | ||||||
|  |         ctx.data().popular_timezones.iter().map(|t| t.to_string()).collect::<Vec<String>>() | ||||||
|  |     } else { | ||||||
|  |         TZ_VARIANTS | ||||||
|  |             .iter() | ||||||
|  |             .filter(|tz| tz.to_string().contains(&partial)) | ||||||
|  |             .take(25) | ||||||
|  |             .map(|t| t.to_string()) | ||||||
|  |             .collect::<Vec<String>>() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | pub async fn macro_name_autocomplete(ctx: Context<'_>, partial: &str) -> Vec<String> { | ||||||
|  |     sqlx::query!( | ||||||
|  |         " | ||||||
|  | SELECT name | ||||||
|  | FROM macro | ||||||
|  | WHERE | ||||||
|  |     guild_id = (SELECT id FROM guilds WHERE guild = ?) | ||||||
|  |     AND name LIKE CONCAT(?, '%')", | ||||||
|  |         ctx.guild_id().unwrap().0, | ||||||
|  |         partial, | ||||||
|  |     ) | ||||||
|  |     .fetch_all(&ctx.data().database) | ||||||
|  |     .await | ||||||
|  |     .unwrap_or_default() | ||||||
|  |     .iter() | ||||||
|  |     .map(|s| s.name.clone()) | ||||||
|  |     .collect() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | pub async fn time_hint_autocomplete( | ||||||
|  |     ctx: Context<'_>, | ||||||
|  |     partial: &str, | ||||||
|  | ) -> Vec<AutocompleteChoice<String>> { | ||||||
|  |     if partial.is_empty() { | ||||||
|  |         vec![AutocompleteChoice { | ||||||
|  |             name: "Start typing a time...".to_string(), | ||||||
|  |             value: "now".to_string(), | ||||||
|  |         }] | ||||||
|  |     } else { | ||||||
|  |         match natural_parser(partial, &ctx.timezone().await.to_string()).await { | ||||||
|  |             Some(timestamp) => match SystemTime::now().duration_since(UNIX_EPOCH) { | ||||||
|  |                 Ok(now) => { | ||||||
|  |                     let diff = timestamp - now.as_secs() as i64; | ||||||
|  |  | ||||||
|  |                     if diff < 0 { | ||||||
|  |                         vec![AutocompleteChoice { | ||||||
|  |                             name: "Time is in the past".to_string(), | ||||||
|  |                             value: "1 year ago".to_string(), | ||||||
|  |                         }] | ||||||
|  |                     } else { | ||||||
|  |                         if diff > 86400 { | ||||||
|  |                             vec![ | ||||||
|  |                                 AutocompleteChoice { | ||||||
|  |                                     name: partial.to_string(), | ||||||
|  |                                     value: partial.to_string(), | ||||||
|  |                                 }, | ||||||
|  |                                 AutocompleteChoice { | ||||||
|  |                                     name: format!( | ||||||
|  |                                         "In approximately {} days, {} hours", | ||||||
|  |                                         diff / 86400, | ||||||
|  |                                         (diff % 86400) / 3600 | ||||||
|  |                                     ), | ||||||
|  |                                     value: partial.to_string(), | ||||||
|  |                                 }, | ||||||
|  |                             ] | ||||||
|  |                         } else if diff > 3600 { | ||||||
|  |                             vec![ | ||||||
|  |                                 AutocompleteChoice { | ||||||
|  |                                     name: partial.to_string(), | ||||||
|  |                                     value: partial.to_string(), | ||||||
|  |                                 }, | ||||||
|  |                                 AutocompleteChoice { | ||||||
|  |                                     name: format!("In approximately {} hours", diff / 3600), | ||||||
|  |                                     value: partial.to_string(), | ||||||
|  |                                 }, | ||||||
|  |                             ] | ||||||
|  |                         } else { | ||||||
|  |                             vec![ | ||||||
|  |                                 AutocompleteChoice { | ||||||
|  |                                     name: partial.to_string(), | ||||||
|  |                                     value: partial.to_string(), | ||||||
|  |                                 }, | ||||||
|  |                                 AutocompleteChoice { | ||||||
|  |                                     name: format!("In approximately {} minutes", diff / 60), | ||||||
|  |                                     value: partial.to_string(), | ||||||
|  |                                 }, | ||||||
|  |                             ] | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 Err(_) => { | ||||||
|  |                     vec![AutocompleteChoice { | ||||||
|  |                         name: partial.to_string(), | ||||||
|  |                         value: partial.to_string(), | ||||||
|  |                     }] | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |  | ||||||
|  |             None => { | ||||||
|  |                 vec![AutocompleteChoice { | ||||||
|  |                     name: "Time not recognised".to_string(), | ||||||
|  |                     value: "now".to_string(), | ||||||
|  |                 }] | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										46
									
								
								src/commands/command_macro/delete.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,46 @@ | |||||||
|  | use super::super::autocomplete::macro_name_autocomplete; | ||||||
|  | use crate::{Context, Error}; | ||||||
|  |  | ||||||
|  | /// Delete a recorded macro | ||||||
|  | #[poise::command( | ||||||
|  |     slash_command, | ||||||
|  |     rename = "delete", | ||||||
|  |     guild_only = true, | ||||||
|  |     default_member_permissions = "MANAGE_GUILD", | ||||||
|  |     identifying_name = "delete_macro" | ||||||
|  | )] | ||||||
|  | pub async fn delete_macro( | ||||||
|  |     ctx: Context<'_>, | ||||||
|  |     #[description = "Name of macro to delete"] | ||||||
|  |     #[autocomplete = "macro_name_autocomplete"] | ||||||
|  |     name: String, | ||||||
|  | ) -> Result<(), Error> { | ||||||
|  |     match sqlx::query!( | ||||||
|  |         " | ||||||
|  | SELECT id FROM macro WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?) AND name = ?", | ||||||
|  |         ctx.guild_id().unwrap().0, | ||||||
|  |         name | ||||||
|  |     ) | ||||||
|  |     .fetch_one(&ctx.data().database) | ||||||
|  |     .await | ||||||
|  |     { | ||||||
|  |         Ok(row) => { | ||||||
|  |             sqlx::query!("DELETE FROM macro WHERE id = ?", row.id) | ||||||
|  |                 .execute(&ctx.data().database) | ||||||
|  |                 .await | ||||||
|  |                 .unwrap(); | ||||||
|  |  | ||||||
|  |             ctx.say(format!("Macro \"{}\" deleted", name)).await?; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         Err(sqlx::Error::RowNotFound) => { | ||||||
|  |             ctx.say(format!("Macro \"{}\" not found", name)).await?; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         Err(e) => { | ||||||
|  |             panic!("{}", e); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
							
								
								
									
										89
									
								
								src/commands/command_macro/list.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,89 @@ | |||||||
|  | use poise::CreateReply; | ||||||
|  |  | ||||||
|  | use crate::{ | ||||||
|  |     component_models::pager::{MacroPager, Pager}, | ||||||
|  |     consts::THEME_COLOR, | ||||||
|  |     models::{command_macro::CommandMacro, CtxData}, | ||||||
|  |     Context, Error, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /// List recorded macros | ||||||
|  | #[poise::command( | ||||||
|  |     slash_command, | ||||||
|  |     rename = "list", | ||||||
|  |     guild_only = true, | ||||||
|  |     default_member_permissions = "MANAGE_GUILD", | ||||||
|  |     identifying_name = "list_macro" | ||||||
|  | )] | ||||||
|  | pub async fn list_macro(ctx: Context<'_>) -> Result<(), Error> { | ||||||
|  |     let macros = ctx.command_macros().await?; | ||||||
|  |  | ||||||
|  |     let resp = show_macro_page(¯os, 0); | ||||||
|  |  | ||||||
|  |     ctx.send(|m| { | ||||||
|  |         *m = resp; | ||||||
|  |         m | ||||||
|  |     }) | ||||||
|  |     .await?; | ||||||
|  |  | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | pub fn max_macro_page<U, E>(macros: &[CommandMacro<U, E>]) -> usize { | ||||||
|  |     ((macros.len() as f64) / 25.0).ceil() as usize | ||||||
|  | } | ||||||
|  |  | ||||||
|  | pub fn show_macro_page<U, E>(macros: &[CommandMacro<U, E>], page: usize) -> CreateReply { | ||||||
|  |     let pager = MacroPager::new(page); | ||||||
|  |  | ||||||
|  |     if macros.is_empty() { | ||||||
|  |         let mut reply = CreateReply::default(); | ||||||
|  |  | ||||||
|  |         reply.embed(|e| { | ||||||
|  |             e.title("Macros") | ||||||
|  |                 .description("No Macros Set Up. Use `/macro record` to get started.") | ||||||
|  |                 .color(*THEME_COLOR) | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         return reply; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let pages = max_macro_page(macros); | ||||||
|  |  | ||||||
|  |     let mut page = page; | ||||||
|  |     if page >= pages { | ||||||
|  |         page = pages - 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let lower = (page * 25).min(macros.len()); | ||||||
|  |     let upper = ((page + 1) * 25).min(macros.len()); | ||||||
|  |  | ||||||
|  |     let fields = macros[lower..upper].iter().map(|m| { | ||||||
|  |         if let Some(description) = &m.description { | ||||||
|  |             ( | ||||||
|  |                 m.name.clone(), | ||||||
|  |                 format!("*{}*\n- Has {} commands", description, m.commands.len()), | ||||||
|  |                 true, | ||||||
|  |             ) | ||||||
|  |         } else { | ||||||
|  |             (m.name.clone(), format!("- Has {} commands", m.commands.len()), true) | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     let mut reply = CreateReply::default(); | ||||||
|  |  | ||||||
|  |     reply | ||||||
|  |         .embed(|e| { | ||||||
|  |             e.title("Macros") | ||||||
|  |                 .fields(fields) | ||||||
|  |                 .footer(|f| f.text(format!("Page {} of {}", page + 1, pages))) | ||||||
|  |                 .color(*THEME_COLOR) | ||||||
|  |         }) | ||||||
|  |         .components(|comp| { | ||||||
|  |             pager.create_button_row(pages, comp); | ||||||
|  |  | ||||||
|  |             comp | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |     reply | ||||||
|  | } | ||||||
							
								
								
									
										229
									
								
								src/commands/command_macro/migrate.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,229 @@ | |||||||
|  | use lazy_regex::regex; | ||||||
|  | use poise::serenity_prelude::command::CommandOptionType; | ||||||
|  | use regex::Captures; | ||||||
|  | use serde_json::{json, Value}; | ||||||
|  |  | ||||||
|  | use crate::{models::command_macro::RawCommandMacro, Context, Error, GuildId}; | ||||||
|  |  | ||||||
|  | struct Alias { | ||||||
|  |     name: String, | ||||||
|  |     command: String, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// Migrate old $alias reminder commands to macros. Only macro names that are not taken will be used. | ||||||
|  | #[poise::command( | ||||||
|  |     slash_command, | ||||||
|  |     rename = "migrate", | ||||||
|  |     guild_only = true, | ||||||
|  |     default_member_permissions = "MANAGE_GUILD", | ||||||
|  |     identifying_name = "migrate_macro" | ||||||
|  | )] | ||||||
|  | pub async fn migrate_macro(ctx: Context<'_>) -> Result<(), Error> { | ||||||
|  |     let guild_id = ctx.guild_id().unwrap(); | ||||||
|  |     let mut transaction = ctx.data().database.begin().await?; | ||||||
|  |  | ||||||
|  |     let aliases = sqlx::query_as!( | ||||||
|  |         Alias, | ||||||
|  |         "SELECT name, command FROM command_aliases WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?)", | ||||||
|  |         guild_id.0 | ||||||
|  |     ) | ||||||
|  |     .fetch_all(&mut *transaction) | ||||||
|  |     .await?; | ||||||
|  |  | ||||||
|  |     let mut added_aliases = 0; | ||||||
|  |  | ||||||
|  |     for alias in aliases { | ||||||
|  |         match parse_text_command(guild_id, alias.name, &alias.command) { | ||||||
|  |             Some(cmd_macro) => { | ||||||
|  |                 sqlx::query!( | ||||||
|  |                     "INSERT INTO macro (guild_id, name, description, commands) VALUES ((SELECT id FROM guilds WHERE guild = ?), ?, ?, ?)", | ||||||
|  |                     cmd_macro.guild_id.0, | ||||||
|  |                     cmd_macro.name, | ||||||
|  |                     cmd_macro.description, | ||||||
|  |                     cmd_macro.commands | ||||||
|  |                 ) | ||||||
|  |                 .execute(&mut *transaction) | ||||||
|  |                 .await?; | ||||||
|  |  | ||||||
|  |                 added_aliases += 1; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             None => {} | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     transaction.commit().await?; | ||||||
|  |  | ||||||
|  |     ctx.send(|b| b.content(format!("Added {} macros.", added_aliases))).await?; | ||||||
|  |  | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn parse_text_command( | ||||||
|  |     guild_id: GuildId, | ||||||
|  |     alias_name: String, | ||||||
|  |     command: &str, | ||||||
|  | ) -> Option<RawCommandMacro> { | ||||||
|  |     match command.split_once(" ") { | ||||||
|  |         Some((command_word, args)) => { | ||||||
|  |             let command_word = command_word.to_lowercase(); | ||||||
|  |  | ||||||
|  |             if command_word == "r" | ||||||
|  |                 || command_word == "i" | ||||||
|  |                 || command_word == "remind" | ||||||
|  |                 || command_word == "interval" | ||||||
|  |             { | ||||||
|  |                 let matcher = regex!( | ||||||
|  |                     r#"(?P<mentions>(?:<@\d+>\s+|<@!\d+>\s+|<#\d+>\s+)*)(?P<time>(?:(?:\d+)(?:s|m|h|d|:|/|-|))+)(?:\s+(?P<interval>(?:(?:\d+)(?:s|m|h|d|))+))?(?:\s+(?P<expires>(?:(?:\d+)(?:s|m|h|d|:|/|-|))+))?\s+(?P<content>.*)"#s | ||||||
|  |                 ); | ||||||
|  |  | ||||||
|  |                 match matcher.captures(&args) { | ||||||
|  |                     Some(captures) => { | ||||||
|  |                         let mut args: Vec<Value> = vec![]; | ||||||
|  |  | ||||||
|  |                         if let Some(group) = captures.name("time") { | ||||||
|  |                             let content = group.as_str(); | ||||||
|  |                             args.push(json!({ | ||||||
|  |                                 "name": "time", | ||||||
|  |                                 "value": content, | ||||||
|  |                                 "type": CommandOptionType::String, | ||||||
|  |                             })); | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         if let Some(group) = captures.name("content") { | ||||||
|  |                             let content = group.as_str(); | ||||||
|  |                             args.push(json!({ | ||||||
|  |                                 "name": "content", | ||||||
|  |                                 "value": content, | ||||||
|  |                                 "type": CommandOptionType::String, | ||||||
|  |                             })); | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         if let Some(group) = captures.name("interval") { | ||||||
|  |                             let content = group.as_str(); | ||||||
|  |                             args.push(json!({ | ||||||
|  |                                 "name": "interval", | ||||||
|  |                                 "value": content, | ||||||
|  |                                 "type": CommandOptionType::String, | ||||||
|  |                             })); | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         if let Some(group) = captures.name("expires") { | ||||||
|  |                             let content = group.as_str(); | ||||||
|  |                             args.push(json!({ | ||||||
|  |                                 "name": "expires", | ||||||
|  |                                 "value": content, | ||||||
|  |                                 "type": CommandOptionType::String, | ||||||
|  |                             })); | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         if let Some(group) = captures.name("mentions") { | ||||||
|  |                             let content = group.as_str(); | ||||||
|  |                             args.push(json!({ | ||||||
|  |                                 "name": "channels", | ||||||
|  |                                 "value": content, | ||||||
|  |                                 "type": CommandOptionType::String, | ||||||
|  |                             })); | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         Some(RawCommandMacro { | ||||||
|  |                             guild_id, | ||||||
|  |                             name: alias_name, | ||||||
|  |                             description: None, | ||||||
|  |                             commands: json!([ | ||||||
|  |                                 { | ||||||
|  |                                     "command_name": "remind", | ||||||
|  |                                     "options": args, | ||||||
|  |                                 } | ||||||
|  |                             ]), | ||||||
|  |                         }) | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     None => None, | ||||||
|  |                 } | ||||||
|  |             } else if command_word == "n" || command_word == "natural" { | ||||||
|  |                 let matcher_primary = regex!( | ||||||
|  |                     r#"(?P<time>.*?)(?:\s+)(?:send|say)(?:\s+)(?P<content>.*?)(?:(?:\s+)to(?:\s+)(?P<mentions>((?:<@\d+>)|(?:<@!\d+>)|(?:<#\d+>)|(?:\s+))+))?$"#s | ||||||
|  |                 ); | ||||||
|  |                 let matcher_secondary = regex!( | ||||||
|  |                     r#"(?P<msg>.*)(?:\s+)every(?:\s+)(?P<interval>.*?)(?:(?:\s+)(?:until|for)(?:\s+)(?P<expires>.*?))?$"#s | ||||||
|  |                 ); | ||||||
|  |  | ||||||
|  |                 match matcher_primary.captures(&args) { | ||||||
|  |                     Some(captures) => { | ||||||
|  |                         let captures_secondary = matcher_secondary.captures(&args); | ||||||
|  |  | ||||||
|  |                         let mut args: Vec<Value> = vec![]; | ||||||
|  |  | ||||||
|  |                         if let Some(group) = captures.name("time") { | ||||||
|  |                             let content = group.as_str(); | ||||||
|  |                             args.push(json!({ | ||||||
|  |                                 "name": "time", | ||||||
|  |                                 "value": content, | ||||||
|  |                                 "type": CommandOptionType::String, | ||||||
|  |                             })); | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         if let Some(group) = captures.name("content") { | ||||||
|  |                             let content = group.as_str(); | ||||||
|  |                             args.push(json!({ | ||||||
|  |                                 "name": "content", | ||||||
|  |                                 "value": content, | ||||||
|  |                                 "type": CommandOptionType::String, | ||||||
|  |                             })); | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         if let Some(group) = | ||||||
|  |                             captures_secondary.as_ref().and_then(|c: &Captures| c.name("interval")) | ||||||
|  |                         { | ||||||
|  |                             let content = group.as_str(); | ||||||
|  |                             args.push(json!({ | ||||||
|  |                                 "name": "interval", | ||||||
|  |                                 "value": content, | ||||||
|  |                                 "type": CommandOptionType::String, | ||||||
|  |                             })); | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         if let Some(group) = | ||||||
|  |                             captures_secondary.and_then(|c: Captures| c.name("expires")) | ||||||
|  |                         { | ||||||
|  |                             let content = group.as_str(); | ||||||
|  |                             args.push(json!({ | ||||||
|  |                                 "name": "expires", | ||||||
|  |                                 "value": content, | ||||||
|  |                                 "type": CommandOptionType::String, | ||||||
|  |                             })); | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         if let Some(group) = captures.name("mentions") { | ||||||
|  |                             let content = group.as_str(); | ||||||
|  |                             args.push(json!({ | ||||||
|  |                                 "name": "channels", | ||||||
|  |                                 "value": content, | ||||||
|  |                                 "type": CommandOptionType::String, | ||||||
|  |                             })); | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         Some(RawCommandMacro { | ||||||
|  |                             guild_id, | ||||||
|  |                             name: alias_name, | ||||||
|  |                             description: None, | ||||||
|  |                             commands: json!([ | ||||||
|  |                                 { | ||||||
|  |                                     "command_name": "remind", | ||||||
|  |                                     "options": args, | ||||||
|  |                                 } | ||||||
|  |                             ]), | ||||||
|  |                         }) | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     None => None, | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 None | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         None => None, | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										19
									
								
								src/commands/command_macro/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | |||||||
|  | use crate::{Context, Error}; | ||||||
|  |  | ||||||
|  | pub mod delete; | ||||||
|  | pub mod list; | ||||||
|  | pub mod migrate; | ||||||
|  | pub mod record; | ||||||
|  | pub mod run; | ||||||
|  |  | ||||||
|  | /// Record and replay command sequences | ||||||
|  | #[poise::command( | ||||||
|  |     slash_command, | ||||||
|  |     rename = "macro", | ||||||
|  |     guild_only = true, | ||||||
|  |     default_member_permissions = "MANAGE_GUILD", | ||||||
|  |     identifying_name = "macro_base" | ||||||
|  | )] | ||||||
|  | pub async fn macro_base(_ctx: Context<'_>) -> Result<(), Error> { | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
							
								
								
									
										151
									
								
								src/commands/command_macro/record.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,151 @@ | |||||||
|  | use std::collections::hash_map::Entry; | ||||||
|  |  | ||||||
|  | use crate::{consts::THEME_COLOR, models::command_macro::CommandMacro, Context, Error}; | ||||||
|  |  | ||||||
|  | /// Start recording up to 5 commands to replay | ||||||
|  | #[poise::command( | ||||||
|  |     slash_command, | ||||||
|  |     rename = "record", | ||||||
|  |     guild_only = true, | ||||||
|  |     default_member_permissions = "MANAGE_GUILD", | ||||||
|  |     identifying_name = "record_macro" | ||||||
|  | )] | ||||||
|  | pub async fn record_macro( | ||||||
|  |     ctx: Context<'_>, | ||||||
|  |     #[description = "Name for the new macro"] name: String, | ||||||
|  |     #[description = "Description for the new macro"] description: Option<String>, | ||||||
|  | ) -> Result<(), Error> { | ||||||
|  |     if name.len() > 100 { | ||||||
|  |         ctx.say("Name must be less than 100 characters").await?; | ||||||
|  |  | ||||||
|  |         return Ok(()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if description.as_ref().map_or(0, |d| d.len()) > 100 { | ||||||
|  |         ctx.say("Description must be less than 100 characters").await?; | ||||||
|  |  | ||||||
|  |         return Ok(()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let guild_id = ctx.guild_id().unwrap(); | ||||||
|  |  | ||||||
|  |     let row = sqlx::query!( | ||||||
|  |         " | ||||||
|  | SELECT 1 as _e FROM macro WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?) AND name = ?", | ||||||
|  |         guild_id.0, | ||||||
|  |         name | ||||||
|  |     ) | ||||||
|  |     .fetch_one(&ctx.data().database) | ||||||
|  |     .await; | ||||||
|  |  | ||||||
|  |     if row.is_ok() { | ||||||
|  |         ctx.send(|m| { | ||||||
|  |             m.ephemeral(true).embed(|e| { | ||||||
|  |                 e.title("Unique Name Required") | ||||||
|  |                     .description( | ||||||
|  |                         "A macro already exists under this name. | ||||||
|  | Please select a unique name for your macro.", | ||||||
|  |                     ) | ||||||
|  |                     .color(*THEME_COLOR) | ||||||
|  |             }) | ||||||
|  |         }) | ||||||
|  |         .await?; | ||||||
|  |     } else { | ||||||
|  |         let okay = { | ||||||
|  |             let mut lock = ctx.data().recording_macros.write().await; | ||||||
|  |  | ||||||
|  |             if let Entry::Vacant(e) = lock.entry((guild_id, ctx.author().id)) { | ||||||
|  |                 e.insert(CommandMacro { guild_id, name, description, commands: vec![] }); | ||||||
|  |                 true | ||||||
|  |             } else { | ||||||
|  |                 false | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         if okay { | ||||||
|  |             ctx.send(|m| { | ||||||
|  |                 m.ephemeral(true).embed(|e| { | ||||||
|  |                     e.title("Macro Recording Started") | ||||||
|  |                         .description( | ||||||
|  |                             "Run up to 5 commands, or type `/macro finish` to stop at any point. | ||||||
|  | Any commands ran as part of recording will be inconsequential", | ||||||
|  |                         ) | ||||||
|  |                         .color(*THEME_COLOR) | ||||||
|  |                 }) | ||||||
|  |             }) | ||||||
|  |             .await?; | ||||||
|  |         } else { | ||||||
|  |             ctx.send(|m| { | ||||||
|  |                 m.ephemeral(true).embed(|e| { | ||||||
|  |                     e.title("Macro Already Recording") | ||||||
|  |                         .description( | ||||||
|  |                             "You are already recording a macro in this server. | ||||||
|  | Please use `/macro finish` to end this recording before starting another.", | ||||||
|  |                         ) | ||||||
|  |                         .color(*THEME_COLOR) | ||||||
|  |                 }) | ||||||
|  |             }) | ||||||
|  |             .await?; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// Finish current macro recording | ||||||
|  | #[poise::command( | ||||||
|  |     slash_command, | ||||||
|  |     rename = "finish", | ||||||
|  |     guild_only = true, | ||||||
|  |     default_member_permissions = "MANAGE_GUILD", | ||||||
|  |     identifying_name = "finish_macro" | ||||||
|  | )] | ||||||
|  | pub async fn finish_macro(ctx: Context<'_>) -> Result<(), Error> { | ||||||
|  |     let key = (ctx.guild_id().unwrap(), ctx.author().id); | ||||||
|  |  | ||||||
|  |     { | ||||||
|  |         let lock = ctx.data().recording_macros.read().await; | ||||||
|  |         let contained = lock.get(&key); | ||||||
|  |  | ||||||
|  |         if contained.map_or(true, |cmacro| cmacro.commands.is_empty()) { | ||||||
|  |             ctx.send(|m| { | ||||||
|  |                 m.embed(|e| { | ||||||
|  |                     e.title("No Macro Recorded") | ||||||
|  |                         .description("Use `/macro record` to start recording a macro") | ||||||
|  |                         .color(*THEME_COLOR) | ||||||
|  |                 }) | ||||||
|  |             }) | ||||||
|  |             .await?; | ||||||
|  |         } else { | ||||||
|  |             let command_macro = contained.unwrap(); | ||||||
|  |             let json = serde_json::to_string(&command_macro.commands).unwrap(); | ||||||
|  |  | ||||||
|  |             sqlx::query!( | ||||||
|  |                 "INSERT INTO macro (guild_id, name, description, commands) VALUES ((SELECT id FROM guilds WHERE guild = ?), ?, ?, ?)", | ||||||
|  |                 command_macro.guild_id.0, | ||||||
|  |                 command_macro.name, | ||||||
|  |                 command_macro.description, | ||||||
|  |                 json | ||||||
|  |             ) | ||||||
|  |                 .execute(&ctx.data().database) | ||||||
|  |                 .await | ||||||
|  |                 .unwrap(); | ||||||
|  |  | ||||||
|  |             ctx.send(|m| { | ||||||
|  |                 m.embed(|e| { | ||||||
|  |                     e.title("Macro Recorded") | ||||||
|  |                         .description("Use `/macro run` to execute the macro") | ||||||
|  |                         .color(*THEME_COLOR) | ||||||
|  |                 }) | ||||||
|  |             }) | ||||||
|  |             .await?; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     { | ||||||
|  |         let mut lock = ctx.data().recording_macros.write().await; | ||||||
|  |         lock.remove(&key); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
							
								
								
									
										56
									
								
								src/commands/command_macro/run.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,56 @@ | |||||||
|  | use super::super::autocomplete::macro_name_autocomplete; | ||||||
|  | use crate::{models::command_macro::guild_command_macro, Context, Data, Error, THEME_COLOR}; | ||||||
|  |  | ||||||
|  | /// Run a recorded macro | ||||||
|  | #[poise::command( | ||||||
|  |     slash_command, | ||||||
|  |     rename = "run", | ||||||
|  |     guild_only = true, | ||||||
|  |     default_member_permissions = "MANAGE_GUILD", | ||||||
|  |     identifying_name = "run_macro" | ||||||
|  | )] | ||||||
|  | pub async fn run_macro( | ||||||
|  |     ctx: poise::ApplicationContext<'_, Data, Error>, | ||||||
|  |     #[description = "Name of macro to run"] | ||||||
|  |     #[autocomplete = "macro_name_autocomplete"] | ||||||
|  |     name: String, | ||||||
|  | ) -> Result<(), Error> { | ||||||
|  |     match guild_command_macro(&Context::Application(ctx), &name).await { | ||||||
|  |         Some(command_macro) => { | ||||||
|  |             Context::Application(ctx) | ||||||
|  |                 .send(|b| { | ||||||
|  |                     b.embed(|e| { | ||||||
|  |                         e.title("Running Macro").color(*THEME_COLOR).description(format!( | ||||||
|  |                             "Running macro {} ({} commands)", | ||||||
|  |                             command_macro.name, | ||||||
|  |                             command_macro.commands.len() | ||||||
|  |                         )) | ||||||
|  |                     }) | ||||||
|  |                 }) | ||||||
|  |                 .await?; | ||||||
|  |  | ||||||
|  |             for command in command_macro.commands { | ||||||
|  |                 if let Some(action) = command.action { | ||||||
|  |                     match (action)(poise::ApplicationContext { args: &command.options, ..ctx }) | ||||||
|  |                         .await | ||||||
|  |                     { | ||||||
|  |                         Ok(()) => {} | ||||||
|  |                         Err(e) => { | ||||||
|  |                             println!("{:?}", e); | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } else { | ||||||
|  |                     Context::Application(ctx) | ||||||
|  |                         .say(format!("Command \"{}\" not found", command.command_name)) | ||||||
|  |                         .await?; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         None => { | ||||||
|  |             Context::Application(ctx).say(format!("Macro \"{}\" not found", name)).await?; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
| @@ -6,8 +6,8 @@ use crate::{models::CtxData, Context, Error, THEME_COLOR}; | |||||||
| fn footer( | fn footer( | ||||||
|     ctx: Context<'_>, |     ctx: Context<'_>, | ||||||
| ) -> impl FnOnce(&mut serenity::CreateEmbedFooter) -> &mut serenity::CreateEmbedFooter { | ) -> impl FnOnce(&mut serenity::CreateEmbedFooter) -> &mut serenity::CreateEmbedFooter { | ||||||
|     let shard_count = ctx.discord().cache.shard_count(); |     let shard_count = ctx.serenity_context().cache.shard_count(); | ||||||
|     let shard = ctx.discord().shard_id; |     let shard = ctx.serenity_context().shard_id; | ||||||
|  |  | ||||||
|     move |f| { |     move |f| { | ||||||
|         f.text(format!( |         f.text(format!( | ||||||
| @@ -49,6 +49,7 @@ __Todo Commands__ | |||||||
|  |  | ||||||
| __Setup Commands__ | __Setup Commands__ | ||||||
| `/timezone` - Set your timezone (necessary for `/remind` to work properly) | `/timezone` - Set your timezone (necessary for `/remind` to work properly) | ||||||
|  | `/dm allow/block` - Change your DM settings for reminders. | ||||||
|  |  | ||||||
| __Advanced Commands__ | __Advanced Commands__ | ||||||
| `/macro` - Record and replay command sequences | `/macro` - Record and replay command sequences | ||||||
| @@ -71,7 +72,7 @@ pub async fn info(ctx: Context<'_>) -> Result<(), Error> { | |||||||
|         .send(|m| { |         .send(|m| { | ||||||
|             m.ephemeral(true).embed(|e| { |             m.ephemeral(true).embed(|e| { | ||||||
|                 e.title("Info") |                 e.title("Info") | ||||||
|                     .description(format!( |                     .description( | ||||||
|                         "Help: `/help` |                         "Help: `/help` | ||||||
|  |  | ||||||
| **Welcome to Reminder Bot!** | **Welcome to Reminder Bot!** | ||||||
| @@ -81,7 +82,7 @@ Find me on https://discord.jellywx.com and on https://github.com/JellyWX :) | |||||||
|  |  | ||||||
| Invite the bot: https://invite.reminder-bot.com/ | Invite the bot: https://invite.reminder-bot.com/ | ||||||
| Use our dashboard: https://reminder-bot.com/", | Use our dashboard: https://reminder-bot.com/", | ||||||
|                     )) |                     ) | ||||||
|                     .footer(footer) |                     .footer(footer) | ||||||
|                     .color(*THEME_COLOR) |                     .color(*THEME_COLOR) | ||||||
|             }) |             }) | ||||||
|   | |||||||
| @@ -1,3 +1,5 @@ | |||||||
|  | mod autocomplete; | ||||||
|  | pub mod command_macro; | ||||||
| pub mod info_cmds; | pub mod info_cmds; | ||||||
| pub mod moderation_cmds; | pub mod moderation_cmds; | ||||||
| pub mod reminder_cmds; | pub mod reminder_cmds; | ||||||
|   | |||||||
| @@ -1,30 +1,10 @@ | |||||||
| use chrono::offset::Utc; | use chrono::offset::Utc; | ||||||
| use chrono_tz::{Tz, TZ_VARIANTS}; | use chrono_tz::{Tz, TZ_VARIANTS}; | ||||||
| use levenshtein::levenshtein; | use levenshtein::levenshtein; | ||||||
| use poise::CreateReply; | use log::warn; | ||||||
|  |  | ||||||
| use crate::{ | use super::autocomplete::timezone_autocomplete; | ||||||
|     component_models::pager::{MacroPager, Pager}, | use crate::{consts::THEME_COLOR, models::CtxData, Context, Error}; | ||||||
|     consts::{EMBED_DESCRIPTION_MAX_LENGTH, THEME_COLOR}, |  | ||||||
|     models::{ |  | ||||||
|         command_macro::{guild_command_macro, CommandMacro}, |  | ||||||
|         CtxData, |  | ||||||
|     }, |  | ||||||
|     Context, Data, Error, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| async fn timezone_autocomplete(ctx: Context<'_>, partial: String) -> Vec<String> { |  | ||||||
|     if partial.is_empty() { |  | ||||||
|         ctx.data().popular_timezones.iter().map(|t| t.to_string()).collect::<Vec<String>>() |  | ||||||
|     } else { |  | ||||||
|         TZ_VARIANTS |  | ||||||
|             .iter() |  | ||||||
|             .filter(|tz| tz.to_string().contains(&partial)) |  | ||||||
|             .take(25) |  | ||||||
|             .map(|t| t.to_string()) |  | ||||||
|             .collect::<Vec<String>>() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /// Select your timezone | /// Select your timezone | ||||||
| #[poise::command(slash_command, identifying_name = "timezone")] | #[poise::command(slash_command, identifying_name = "timezone")] | ||||||
| @@ -52,7 +32,7 @@ pub async fn timezone( | |||||||
|                             .description(format!( |                             .description(format!( | ||||||
|                                 "Timezone has been set to **{}**. Your current time should be `{}`", |                                 "Timezone has been set to **{}**. Your current time should be `{}`", | ||||||
|                                 timezone, |                                 timezone, | ||||||
|                                 now.format("%H:%M").to_string() |                                 now.format("%H:%M") | ||||||
|                             )) |                             )) | ||||||
|                             .color(*THEME_COLOR) |                             .color(*THEME_COLOR) | ||||||
|                     }) |                     }) | ||||||
| @@ -75,10 +55,7 @@ pub async fn timezone( | |||||||
|                 let fields = filtered_tz.iter().map(|tz| { |                 let fields = filtered_tz.iter().map(|tz| { | ||||||
|                     ( |                     ( | ||||||
|                         tz.to_string(), |                         tz.to_string(), | ||||||
|                         format!( |                         format!("🕗 `{}`", Utc::now().with_timezone(tz).format("%H:%M")), | ||||||
|                             "🕗 `{}`", |  | ||||||
|                             Utc::now().with_timezone(tz).format("%H:%M").to_string() |  | ||||||
|                         ), |  | ||||||
|                         true, |                         true, | ||||||
|                     ) |                     ) | ||||||
|                 }); |                 }); | ||||||
| @@ -98,11 +75,7 @@ pub async fn timezone( | |||||||
|         } |         } | ||||||
|     } else { |     } else { | ||||||
|         let popular_timezones_iter = ctx.data().popular_timezones.iter().map(|t| { |         let popular_timezones_iter = ctx.data().popular_timezones.iter().map(|t| { | ||||||
|             ( |             (t.to_string(), format!("🕗 `{}`", Utc::now().with_timezone(t).format("%H:%M")), true) | ||||||
|                 t.to_string(), |  | ||||||
|                 format!("🕗 `{}`", Utc::now().with_timezone(t).format("%H:%M").to_string()), |  | ||||||
|                 true, |  | ||||||
|             ) |  | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
|         ctx.send(|m| { |         ctx.send(|m| { | ||||||
| @@ -129,379 +102,154 @@ You may want to use one of the popular timezones below, otherwise click [here](h | |||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
|  |  | ||||||
| async fn macro_name_autocomplete(ctx: Context<'_>, partial: String) -> Vec<String> { | /// Configure server settings | ||||||
|     sqlx::query!( |  | ||||||
|         " |  | ||||||
| SELECT name |  | ||||||
| FROM macro |  | ||||||
| WHERE |  | ||||||
|     guild_id = (SELECT id FROM guilds WHERE guild = ?) |  | ||||||
|     AND name LIKE CONCAT(?, '%')", |  | ||||||
|         ctx.guild_id().unwrap().0, |  | ||||||
|         partial, |  | ||||||
|     ) |  | ||||||
|     .fetch_all(&ctx.data().database) |  | ||||||
|     .await |  | ||||||
|     .unwrap_or(vec![]) |  | ||||||
|     .iter() |  | ||||||
|     .map(|s| s.name.clone()) |  | ||||||
|     .collect() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /// Record and replay command sequences |  | ||||||
| #[poise::command( | #[poise::command( | ||||||
|     slash_command, |     slash_command, | ||||||
|     rename = "macro", |     rename = "settings", | ||||||
|     guild_only = true, |     identifying_name = "settings", | ||||||
|     default_member_permissions = "MANAGE_GUILD", |     guild_only = true | ||||||
|     identifying_name = "macro_base" |  | ||||||
| )] | )] | ||||||
| pub async fn macro_base(_ctx: Context<'_>) -> Result<(), Error> { | pub async fn settings(_ctx: Context<'_>) -> Result<(), Error> { | ||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
|  |  | ||||||
| /// Start recording up to 5 commands to replay | /// Configure ephemeral setup | ||||||
| #[poise::command( | #[poise::command( | ||||||
|     slash_command, |     slash_command, | ||||||
|     rename = "record", |     rename = "ephemeral", | ||||||
|     guild_only = true, |     identifying_name = "ephemeral_confirmations", | ||||||
|     default_member_permissions = "MANAGE_GUILD", |     guild_only = true | ||||||
|     identifying_name = "record_macro" |  | ||||||
| )] | )] | ||||||
| pub async fn record_macro( | pub async fn ephemeral_confirmations(_ctx: Context<'_>) -> Result<(), Error> { | ||||||
|     ctx: Context<'_>, |     Ok(()) | ||||||
|     #[description = "Name for the new macro"] name: String, | } | ||||||
|     #[description = "Description for the new macro"] description: Option<String>, |  | ||||||
| ) -> Result<(), Error> { |  | ||||||
|     let guild_id = ctx.guild_id().unwrap(); |  | ||||||
|  |  | ||||||
|     let row = sqlx::query!( | /// Set reminder confirmations to be sent "ephemerally" (private and cleared automatically) | ||||||
|         " | #[poise::command( | ||||||
| SELECT 1 as _e FROM macro WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?) AND name = ?", |     slash_command, | ||||||
|         guild_id.0, |     rename = "on", | ||||||
|         name |     identifying_name = "set_ephemeral_confirmations", | ||||||
|     ) |     guild_only = true | ||||||
|     .fetch_one(&ctx.data().database) | )] | ||||||
|     .await; | pub async fn set_ephemeral_confirmations(ctx: Context<'_>) -> Result<(), Error> { | ||||||
|  |     let mut guild_data = ctx.guild_data().await.unwrap()?; | ||||||
|  |     guild_data.ephemeral_confirmations = true; | ||||||
|  |     guild_data.commit_changes(&ctx.data().database).await; | ||||||
|  |  | ||||||
|     if row.is_ok() { |     ctx.send(|r| { | ||||||
|         ctx.send(|m| { |         r.ephemeral(true).embed(|e| { | ||||||
|             m.ephemeral(true).embed(|e| { |             e.title("Confirmations ephemeral") | ||||||
|                 e.title("Unique Name Required") |                 .description("Reminder confirmations will be sent privately, and removed when your client restarts.") | ||||||
|                     .description( |                 .color(*THEME_COLOR) | ||||||
|                         "A macro already exists under this name. |  | ||||||
| Please select a unique name for your macro.", |  | ||||||
|                     ) |  | ||||||
|                     .color(*THEME_COLOR) |  | ||||||
|             }) |  | ||||||
|         }) |         }) | ||||||
|         .await?; |  | ||||||
|     } else { |  | ||||||
|         let okay = { |  | ||||||
|             let mut lock = ctx.data().recording_macros.write().await; |  | ||||||
|  |  | ||||||
|             if lock.contains_key(&(guild_id, ctx.author().id)) { |  | ||||||
|                 false |  | ||||||
|             } else { |  | ||||||
|                 lock.insert( |  | ||||||
|                     (guild_id, ctx.author().id), |  | ||||||
|                     CommandMacro { guild_id, name, description, commands: vec![] }, |  | ||||||
|                 ); |  | ||||||
|                 true |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         if okay { |  | ||||||
|             ctx.send(|m| { |  | ||||||
|                 m.ephemeral(true).embed(|e| { |  | ||||||
|                     e.title("Macro Recording Started") |  | ||||||
|                         .description( |  | ||||||
|                             "Run up to 5 commands, or type `/macro finish` to stop at any point. |  | ||||||
| Any commands ran as part of recording will be inconsequential", |  | ||||||
|                         ) |  | ||||||
|                         .color(*THEME_COLOR) |  | ||||||
|                 }) |  | ||||||
|             }) |  | ||||||
|             .await?; |  | ||||||
|         } else { |  | ||||||
|             ctx.send(|m| { |  | ||||||
|                 m.ephemeral(true).embed(|e| { |  | ||||||
|                     e.title("Macro Already Recording") |  | ||||||
|                         .description( |  | ||||||
|                             "You are already recording a macro in this server. |  | ||||||
| Please use `/macro finish` to end this recording before starting another.", |  | ||||||
|                         ) |  | ||||||
|                         .color(*THEME_COLOR) |  | ||||||
|                 }) |  | ||||||
|             }) |  | ||||||
|             .await?; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     Ok(()) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /// Finish current macro recording |  | ||||||
| #[poise::command( |  | ||||||
|     slash_command, |  | ||||||
|     rename = "finish", |  | ||||||
|     guild_only = true, |  | ||||||
|     default_member_permissions = "MANAGE_GUILD", |  | ||||||
|     identifying_name = "finish_macro" |  | ||||||
| )] |  | ||||||
| pub async fn finish_macro(ctx: Context<'_>) -> Result<(), Error> { |  | ||||||
|     let key = (ctx.guild_id().unwrap(), ctx.author().id); |  | ||||||
|  |  | ||||||
|     { |  | ||||||
|         let lock = ctx.data().recording_macros.read().await; |  | ||||||
|         let contained = lock.get(&key); |  | ||||||
|  |  | ||||||
|         if contained.map_or(true, |cmacro| cmacro.commands.is_empty()) { |  | ||||||
|             ctx.send(|m| { |  | ||||||
|                 m.embed(|e| { |  | ||||||
|                     e.title("No Macro Recorded") |  | ||||||
|                         .description("Use `/macro record` to start recording a macro") |  | ||||||
|                         .color(*THEME_COLOR) |  | ||||||
|                 }) |  | ||||||
|             }) |  | ||||||
|             .await?; |  | ||||||
|         } else { |  | ||||||
|             let command_macro = contained.unwrap(); |  | ||||||
|             let json = serde_json::to_string(&command_macro.commands).unwrap(); |  | ||||||
|  |  | ||||||
|             sqlx::query!( |  | ||||||
|                 "INSERT INTO macro (guild_id, name, description, commands) VALUES ((SELECT id FROM guilds WHERE guild = ?), ?, ?, ?)", |  | ||||||
|                 command_macro.guild_id.0, |  | ||||||
|                 command_macro.name, |  | ||||||
|                 command_macro.description, |  | ||||||
|                 json |  | ||||||
|             ) |  | ||||||
|                 .execute(&ctx.data().database) |  | ||||||
|                 .await |  | ||||||
|                 .unwrap(); |  | ||||||
|  |  | ||||||
|             ctx.send(|m| { |  | ||||||
|                 m.embed(|e| { |  | ||||||
|                     e.title("Macro Recorded") |  | ||||||
|                         .description("Use `/macro run` to execute the macro") |  | ||||||
|                         .color(*THEME_COLOR) |  | ||||||
|                 }) |  | ||||||
|             }) |  | ||||||
|             .await?; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     { |  | ||||||
|         let mut lock = ctx.data().recording_macros.write().await; |  | ||||||
|         lock.remove(&key); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     Ok(()) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /// List recorded macros |  | ||||||
| #[poise::command( |  | ||||||
|     slash_command, |  | ||||||
|     rename = "list", |  | ||||||
|     guild_only = true, |  | ||||||
|     default_member_permissions = "MANAGE_GUILD", |  | ||||||
|     identifying_name = "list_macro" |  | ||||||
| )] |  | ||||||
| pub async fn list_macro(ctx: Context<'_>) -> Result<(), Error> { |  | ||||||
|     let macros = ctx.command_macros().await?; |  | ||||||
|  |  | ||||||
|     let resp = show_macro_page(¯os, 0); |  | ||||||
|  |  | ||||||
|     ctx.send(|m| { |  | ||||||
|         *m = resp; |  | ||||||
|         m |  | ||||||
|     }) |     }) | ||||||
|     .await?; |     .await?; | ||||||
|  |  | ||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
|  |  | ||||||
| /// Run a recorded macro | /// Set reminder confirmations to persist indefinitely | ||||||
| #[poise::command( | #[poise::command( | ||||||
|     slash_command, |     slash_command, | ||||||
|     rename = "run", |     rename = "off", | ||||||
|     guild_only = true, |     identifying_name = "unset_ephemeral_confirmations", | ||||||
|     default_member_permissions = "MANAGE_GUILD", |     guild_only = true | ||||||
|     identifying_name = "run_macro" |  | ||||||
| )] | )] | ||||||
| pub async fn run_macro( | pub async fn unset_ephemeral_confirmations(ctx: Context<'_>) -> Result<(), Error> { | ||||||
|     ctx: poise::ApplicationContext<'_, Data, Error>, |     let mut guild_data = ctx.guild_data().await.unwrap()?; | ||||||
|     #[description = "Name of macro to run"] |     guild_data.ephemeral_confirmations = false; | ||||||
|     #[autocomplete = "macro_name_autocomplete"] |     guild_data.commit_changes(&ctx.data().database).await; | ||||||
|     name: String, |  | ||||||
| ) -> Result<(), Error> { |  | ||||||
|     match guild_command_macro(&Context::Application(ctx), &name).await { |  | ||||||
|         Some(command_macro) => { |  | ||||||
|             ctx.defer_response(false).await?; |  | ||||||
|  |  | ||||||
|             for command in command_macro.commands { |     ctx.send(|r| { | ||||||
|                 if let Some(action) = command.action { |         r.ephemeral(true).embed(|e| { | ||||||
|                     match (action)(poise::ApplicationContext { args: &command.options, ..ctx }) |             e.title("Confirmations public") | ||||||
|                         .await |                 .description( | ||||||
|                     { |                     "Reminder confirmations will be sent as regular messages, and won't be removed automatically.", | ||||||
|                         Ok(()) => {} |                 ) | ||||||
|                         Err(e) => { |                 .color(*THEME_COLOR) | ||||||
|                             println!("{:?}", e); |         }) | ||||||
|                         } |     }) | ||||||
|                     } |     .await?; | ||||||
|                 } else { |  | ||||||
|                     Context::Application(ctx) |  | ||||||
|                         .say(format!("Command \"{}\" not found", command.command_name)) |  | ||||||
|                         .await?; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         None => { |  | ||||||
|             Context::Application(ctx).say(format!("Macro \"{}\" not found", name)).await?; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
|  |  | ||||||
| /// Delete a recorded macro | /// Configure whether other users can set reminders to your direct messages | ||||||
|  | #[poise::command(slash_command, rename = "dm", identifying_name = "allowed_dm")] | ||||||
|  | pub async fn allowed_dm(_ctx: Context<'_>) -> Result<(), Error> { | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// Allow other users to set reminders in your direct messages | ||||||
|  | #[poise::command(slash_command, rename = "allow", identifying_name = "set_allowed_dm")] | ||||||
|  | pub async fn set_allowed_dm(ctx: Context<'_>) -> Result<(), Error> { | ||||||
|  |     let mut user_data = ctx.author_data().await?; | ||||||
|  |     user_data.allowed_dm = true; | ||||||
|  |     user_data.commit_changes(&ctx.data().database).await; | ||||||
|  |  | ||||||
|  |     ctx.send(|r| { | ||||||
|  |         r.ephemeral(true).embed(|e| { | ||||||
|  |             e.title("DMs permitted") | ||||||
|  |                 .description("You will receive a message if a user sets a DM reminder for you.") | ||||||
|  |                 .color(*THEME_COLOR) | ||||||
|  |         }) | ||||||
|  |     }) | ||||||
|  |     .await?; | ||||||
|  |  | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// Block other users from setting reminders in your direct messages | ||||||
|  | #[poise::command(slash_command, rename = "block", identifying_name = "unset_allowed_dm")] | ||||||
|  | pub async fn unset_allowed_dm(ctx: Context<'_>) -> Result<(), Error> { | ||||||
|  |     let mut user_data = ctx.author_data().await?; | ||||||
|  |     user_data.allowed_dm = false; | ||||||
|  |     user_data.commit_changes(&ctx.data().database).await; | ||||||
|  |  | ||||||
|  |     ctx.send(|r| { | ||||||
|  |         r.ephemeral(true).embed(|e| { | ||||||
|  |             e.title("DMs blocked") | ||||||
|  |                 .description( | ||||||
|  |                     "You can still set DM reminders for yourself or for users with DMs enabled.", | ||||||
|  |                 ) | ||||||
|  |                 .color(*THEME_COLOR) | ||||||
|  |         }) | ||||||
|  |     }) | ||||||
|  |     .await?; | ||||||
|  |  | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// View the webhook being used to send reminders to this channel | ||||||
| #[poise::command( | #[poise::command( | ||||||
|     slash_command, |     slash_command, | ||||||
|     rename = "delete", |     identifying_name = "webhook_url", | ||||||
|     guild_only = true, |     required_permissions = "ADMINISTRATOR" | ||||||
|     default_member_permissions = "MANAGE_GUILD", |  | ||||||
|     identifying_name = "delete_macro" |  | ||||||
| )] | )] | ||||||
| pub async fn delete_macro( | pub async fn webhook(ctx: Context<'_>) -> Result<(), Error> { | ||||||
|     ctx: Context<'_>, |     match ctx.channel_data().await { | ||||||
|     #[description = "Name of macro to delete"] |         Ok(data) => { | ||||||
|     #[autocomplete = "macro_name_autocomplete"] |             if let (Some(id), Some(token)) = (data.webhook_id, data.webhook_token) { | ||||||
|     name: String, |                 ctx.send(|b| { | ||||||
| ) -> Result<(), Error> { |                     b.ephemeral(true).content(format!( | ||||||
|     match sqlx::query!( |                         "**Warning!** | ||||||
|         " | This link can be used by users to anonymously send messages, with or without permissions. | ||||||
| SELECT id FROM macro WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?) AND name = ?", | Do not share it! | ||||||
|         ctx.guild_id().unwrap().0, | || https://discord.com/api/webhooks/{}/{} ||", | ||||||
|         name |                         id, token, | ||||||
|     ) |                     )) | ||||||
|     .fetch_one(&ctx.data().database) |                 }) | ||||||
|     .await |                 .await?; | ||||||
|     { |             } else { | ||||||
|         Ok(row) => { |                 ctx.say("No webhook configured on this channel.").await?; | ||||||
|             sqlx::query!("DELETE FROM macro WHERE id = ?", row.id) |             } | ||||||
|                 .execute(&ctx.data().database) |  | ||||||
|                 .await |  | ||||||
|                 .unwrap(); |  | ||||||
|  |  | ||||||
|             ctx.say(format!("Macro \"{}\" deleted", name)).await?; |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         Err(sqlx::Error::RowNotFound) => { |  | ||||||
|             ctx.say(format!("Macro \"{}\" not found", name)).await?; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Err(e) => { |         Err(e) => { | ||||||
|             panic!("{}", e); |             warn!("Error fetching channel data: {:?}", e); | ||||||
|  |  | ||||||
|  |             ctx.say("No webhook configured on this channel.").await?; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
|  |  | ||||||
| pub fn max_macro_page<U, E>(macros: &[CommandMacro<U, E>]) -> usize { |  | ||||||
|     let mut skipped_char_count = 0; |  | ||||||
|  |  | ||||||
|     macros |  | ||||||
|         .iter() |  | ||||||
|         .map(|m| { |  | ||||||
|             if let Some(description) = &m.description { |  | ||||||
|                 format!("**{}**\n- *{}*\n- Has {} commands", m.name, description, m.commands.len()) |  | ||||||
|             } else { |  | ||||||
|                 format!("**{}**\n- Has {} commands", m.name, m.commands.len()) |  | ||||||
|             } |  | ||||||
|         }) |  | ||||||
|         .fold(1, |mut pages, p| { |  | ||||||
|             skipped_char_count += p.len(); |  | ||||||
|  |  | ||||||
|             if skipped_char_count > EMBED_DESCRIPTION_MAX_LENGTH { |  | ||||||
|                 skipped_char_count = p.len(); |  | ||||||
|                 pages += 1; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             pages |  | ||||||
|         }) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| pub fn show_macro_page<U, E>(macros: &[CommandMacro<U, E>], page: usize) -> CreateReply { |  | ||||||
|     let pager = MacroPager::new(page); |  | ||||||
|  |  | ||||||
|     if macros.is_empty() { |  | ||||||
|         let mut reply = CreateReply::default(); |  | ||||||
|  |  | ||||||
|         reply.embed(|e| { |  | ||||||
|             e.title("Macros") |  | ||||||
|                 .description("No Macros Set Up. Use `/macro record` to get started.") |  | ||||||
|                 .color(*THEME_COLOR) |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         return reply; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     let pages = max_macro_page(macros); |  | ||||||
|  |  | ||||||
|     let mut page = page; |  | ||||||
|     if page >= pages { |  | ||||||
|         page = pages - 1; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     let mut char_count = 0; |  | ||||||
|     let mut skipped_char_count = 0; |  | ||||||
|  |  | ||||||
|     let mut skipped_pages = 0; |  | ||||||
|  |  | ||||||
|     let display_vec: Vec<String> = macros |  | ||||||
|         .iter() |  | ||||||
|         .map(|m| { |  | ||||||
|             if let Some(description) = &m.description { |  | ||||||
|                 format!("**{}**\n- *{}*\n- Has {} commands", m.name, description, m.commands.len()) |  | ||||||
|             } else { |  | ||||||
|                 format!("**{}**\n- Has {} commands", m.name, m.commands.len()) |  | ||||||
|             } |  | ||||||
|         }) |  | ||||||
|         .skip_while(|p| { |  | ||||||
|             skipped_char_count += p.len(); |  | ||||||
|  |  | ||||||
|             if skipped_char_count > EMBED_DESCRIPTION_MAX_LENGTH { |  | ||||||
|                 skipped_char_count = p.len(); |  | ||||||
|                 skipped_pages += 1; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             skipped_pages < page |  | ||||||
|         }) |  | ||||||
|         .take_while(|p| { |  | ||||||
|             char_count += p.len(); |  | ||||||
|  |  | ||||||
|             char_count < EMBED_DESCRIPTION_MAX_LENGTH |  | ||||||
|         }) |  | ||||||
|         .collect::<Vec<String>>(); |  | ||||||
|  |  | ||||||
|     let display = display_vec.join("\n"); |  | ||||||
|  |  | ||||||
|     let mut reply = CreateReply::default(); |  | ||||||
|  |  | ||||||
|     reply |  | ||||||
|         .embed(|e| { |  | ||||||
|             e.title("Macros") |  | ||||||
|                 .description(display) |  | ||||||
|                 .footer(|f| f.text(format!("Page {} of {}", page + 1, pages))) |  | ||||||
|                 .color(*THEME_COLOR) |  | ||||||
|         }) |  | ||||||
|         .components(|comp| { |  | ||||||
|             pager.create_button_row(pages, comp); |  | ||||||
|  |  | ||||||
|             comp |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|     reply |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -1,21 +1,21 @@ | |||||||
| use std::{ | use std::{collections::HashSet, string::ToString}; | ||||||
|     collections::HashSet, |  | ||||||
|     string::ToString, |  | ||||||
|     time::{SystemTime, UNIX_EPOCH}, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| use chrono::NaiveDateTime; | use chrono::{DateTime, NaiveDateTime, Utc}; | ||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
|  | use log::warn; | ||||||
| use num_integer::Integer; | use num_integer::Integer; | ||||||
| use poise::{ | use poise::{ | ||||||
|     serenity::{builder::CreateEmbed, model::channel::Channel}, |     serenity_prelude::{ | ||||||
|     CreateReply, |         builder::CreateEmbed, component::ButtonStyle, model::channel::Channel, ReactionType, | ||||||
|  |     }, | ||||||
|  |     CreateReply, Modal, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
|  |     commands::autocomplete::{time_hint_autocomplete, timezone_autocomplete}, | ||||||
|     component_models::{ |     component_models::{ | ||||||
|         pager::{DelPager, LookPager, Pager}, |         pager::{DelPager, LookPager, Pager}, | ||||||
|         ComponentDataModel, DelSelector, |         ComponentDataModel, DelSelector, UndoReminder, | ||||||
|     }, |     }, | ||||||
|     consts::{ |     consts::{ | ||||||
|         EMBED_DESCRIPTION_MAX_LENGTH, HOUR, MINUTE, REGEX_CHANNEL_USER, SELECT_MAX_ENTRIES, |         EMBED_DESCRIPTION_MAX_LENGTH, HOUR, MINUTE, REGEX_CHANNEL_USER, SELECT_MAX_ENTRIES, | ||||||
| @@ -35,7 +35,7 @@ use crate::{ | |||||||
|     }, |     }, | ||||||
|     time_parser::natural_parser, |     time_parser::natural_parser, | ||||||
|     utils::{check_guild_subscription, check_subscription}, |     utils::{check_guild_subscription, check_subscription}, | ||||||
|     Context, Error, |     ApplicationContext, Context, Error, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| /// Pause all reminders on the current channel until a certain time or indefinitely | /// Pause all reminders on the current channel until a certain time or indefinitely | ||||||
| @@ -57,18 +57,27 @@ pub async fn pause( | |||||||
|             let parsed = natural_parser(&until, &timezone.to_string()).await; |             let parsed = natural_parser(&until, &timezone.to_string()).await; | ||||||
|  |  | ||||||
|             if let Some(timestamp) = parsed { |             if let Some(timestamp) = parsed { | ||||||
|                 let dt = NaiveDateTime::from_timestamp(timestamp, 0); |                 match NaiveDateTime::from_timestamp_opt(timestamp, 0) { | ||||||
|  |                     Some(dt) => { | ||||||
|  |                         channel.paused = true; | ||||||
|  |                         channel.paused_until = Some(dt); | ||||||
|  |  | ||||||
|                 channel.paused = true; |                         channel.commit_changes(&ctx.data().database).await; | ||||||
|                 channel.paused_until = Some(dt); |  | ||||||
|  |  | ||||||
|                 channel.commit_changes(&ctx.data().database).await; |                         ctx.say(format!( | ||||||
|  |                             "Reminders in this channel have been silenced until **<t:{}:D>**", | ||||||
|  |                             timestamp | ||||||
|  |                         )) | ||||||
|  |                         .await?; | ||||||
|  |                     } | ||||||
|  |  | ||||||
|                 ctx.say(format!( |                     None => { | ||||||
|                     "Reminders in this channel have been silenced until **<t:{}:D>**", |                         ctx.say(format!( | ||||||
|                     timestamp |                             "Time processed could not be interpreted as `DateTime`. Please write the time as clearly as possible", | ||||||
|                 )) |                         )) | ||||||
|                 .await?; |                         .await?; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|             } else { |             } else { | ||||||
|                 ctx.say( |                 ctx.say( | ||||||
|                     "Time could not be processed. Please write the time as clearly as possible", |                     "Time could not be processed. Please write the time as clearly as possible", | ||||||
| @@ -105,6 +114,8 @@ pub async fn offset( | |||||||
|     #[description = "Number of minutes to offset by"] minutes: Option<isize>, |     #[description = "Number of minutes to offset by"] minutes: Option<isize>, | ||||||
|     #[description = "Number of seconds to offset by"] seconds: Option<isize>, |     #[description = "Number of seconds to offset by"] seconds: Option<isize>, | ||||||
| ) -> Result<(), Error> { | ) -> Result<(), Error> { | ||||||
|  |     ctx.defer().await?; | ||||||
|  |  | ||||||
|     let combined_time = hours.map_or(0, |h| h * HOUR as isize) |     let combined_time = hours.map_or(0, |h| h * HOUR as isize) | ||||||
|         + minutes.map_or(0, |m| m * MINUTE as isize) |         + minutes.map_or(0, |m| m * MINUTE as isize) | ||||||
|         + seconds.map_or(0, |s| s); |         + seconds.map_or(0, |s| s); | ||||||
| @@ -207,7 +218,7 @@ pub async fn look( | |||||||
|         }), |         }), | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     let channel_opt = ctx.channel_id().to_channel_cached(&ctx.discord()); |     let channel_opt = ctx.channel_id().to_channel_cached(&ctx); | ||||||
|  |  | ||||||
|     let channel_id = if let Some(Channel::Guild(channel)) = channel_opt { |     let channel_id = if let Some(Channel::Guild(channel)) = channel_opt { | ||||||
|         if Some(channel.guild_id) == ctx.guild_id() { |         if Some(channel.guild_id) == ctx.guild_id() { | ||||||
| @@ -219,12 +230,11 @@ pub async fn look( | |||||||
|         ctx.channel_id() |         ctx.channel_id() | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     let channel_name = |     let channel_name = if let Some(Channel::Guild(channel)) = channel_id.to_channel_cached(&ctx) { | ||||||
|         if let Some(Channel::Guild(channel)) = channel_id.to_channel_cached(&ctx.discord()) { |         Some(channel.name) | ||||||
|             Some(channel.name) |     } else { | ||||||
|         } else { |         None | ||||||
|             None |     }; | ||||||
|         }; |  | ||||||
|  |  | ||||||
|     let reminders = Reminder::from_channel(&ctx.data().database, channel_id, &flags).await; |     let reminders = Reminder::from_channel(&ctx.data().database, channel_id, &flags).await; | ||||||
|  |  | ||||||
| @@ -242,7 +252,7 @@ pub async fn look( | |||||||
|                 char_count < EMBED_DESCRIPTION_MAX_LENGTH |                 char_count < EMBED_DESCRIPTION_MAX_LENGTH | ||||||
|             }) |             }) | ||||||
|             .collect::<Vec<String>>() |             .collect::<Vec<String>>() | ||||||
|             .join("\n"); |             .join(""); | ||||||
|  |  | ||||||
|         let pages = reminders |         let pages = reminders | ||||||
|             .iter() |             .iter() | ||||||
| @@ -286,8 +296,7 @@ pub async fn delete(ctx: Context<'_>) -> Result<(), Error> { | |||||||
|     let timezone = ctx.timezone().await; |     let timezone = ctx.timezone().await; | ||||||
|  |  | ||||||
|     let reminders = |     let reminders = | ||||||
|         Reminder::from_guild(&ctx.discord(), &ctx.data().database, ctx.guild_id(), ctx.author().id) |         Reminder::from_guild(&ctx, &ctx.data().database, ctx.guild_id(), ctx.author().id).await; | ||||||
|             .await; |  | ||||||
|  |  | ||||||
|     let resp = show_delete_page(&reminders, 0, timezone); |     let resp = show_delete_page(&reminders, 0, timezone); | ||||||
|  |  | ||||||
| @@ -429,11 +438,8 @@ pub fn show_delete_page(reminders: &[Reminder], page: usize, timezone: Tz) -> Cr | |||||||
|     reply |     reply | ||||||
| } | } | ||||||
|  |  | ||||||
| fn time_difference(start_time: NaiveDateTime) -> String { | fn time_difference(start_time: DateTime<Utc>) -> String { | ||||||
|     let unix_time = SystemTime::now().duration_since(UNIX_EPOCH).unwrap().as_secs() as i64; |     let delta = (Utc::now() - start_time).num_seconds(); | ||||||
|     let now = NaiveDateTime::from_timestamp(unix_time, 0); |  | ||||||
|  |  | ||||||
|     let delta = (now - start_time).num_seconds(); |  | ||||||
|  |  | ||||||
|     let (minutes, seconds) = delta.div_rem(&60); |     let (minutes, seconds) = delta.div_rem(&60); | ||||||
|     let (hours, minutes) = minutes.div_rem(&60); |     let (hours, minutes) = minutes.div_rem(&60); | ||||||
| @@ -500,18 +506,16 @@ pub async fn start_timer( | |||||||
|     if count >= 25 { |     if count >= 25 { | ||||||
|         ctx.say("You already have 25 timers. Please delete some timers before creating a new one") |         ctx.say("You already have 25 timers. Please delete some timers before creating a new one") | ||||||
|             .await?; |             .await?; | ||||||
|     } else { |     } else if name.len() <= 32 { | ||||||
|         if name.len() <= 32 { |         Timer::create(&name, owner, &ctx.data().database).await; | ||||||
|             Timer::create(&name, owner, &ctx.data().database).await; |  | ||||||
|  |  | ||||||
|             ctx.say("Created a new timer").await?; |         ctx.say("Created a new timer").await?; | ||||||
|         } else { |     } else { | ||||||
|             ctx.say(format!( |         ctx.say(format!( | ||||||
|                 "Please name your timer something shorted (max. 32 characters, you used {})", |             "Please name your timer something shorted (max. 32 characters, you used {})", | ||||||
|                 name.len() |             name.len() | ||||||
|             )) |         )) | ||||||
|             .await?; |         .await?; | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     Ok(()) |     Ok(()) | ||||||
| @@ -549,23 +553,104 @@ pub async fn delete_timer( | |||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
|  |  | ||||||
| /// Create a new reminder | #[derive(poise::Modal)] | ||||||
|  | #[name = "Reminder"] | ||||||
|  | struct ContentModal { | ||||||
|  |     #[name = "Content"] | ||||||
|  |     #[placeholder = "Message..."] | ||||||
|  |     #[paragraph] | ||||||
|  |     #[max_length = 2000] | ||||||
|  |     content: String, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// Create a reminder with multi-line content. Press "+4 more" for other options. | ||||||
|  | #[poise::command( | ||||||
|  |     slash_command, | ||||||
|  |     identifying_name = "multiline", | ||||||
|  |     default_member_permissions = "MANAGE_GUILD" | ||||||
|  | )] | ||||||
|  | pub async fn multiline( | ||||||
|  |     ctx: ApplicationContext<'_>, | ||||||
|  |     #[description = "A description of the time to set the reminder for"] | ||||||
|  |     #[autocomplete = "time_hint_autocomplete"] | ||||||
|  |     time: String, | ||||||
|  |     #[description = "Channel or user mentions to set the reminder for"] channels: Option<String>, | ||||||
|  |     #[description = "(Patreon only) Time to wait before repeating the reminder. Leave blank for one-shot reminder"] | ||||||
|  |     interval: Option<String>, | ||||||
|  |     #[description = "(Patreon only) For repeating reminders, the time at which the reminder will stop repeating"] | ||||||
|  |     expires: Option<String>, | ||||||
|  |     #[description = "Set the TTS flag on the reminder message, similar to the /tts command"] | ||||||
|  |     tts: Option<bool>, | ||||||
|  |     #[description = "Set a timezone override for this reminder only"] | ||||||
|  |     #[autocomplete = "timezone_autocomplete"] | ||||||
|  |     timezone: Option<String>, | ||||||
|  | ) -> Result<(), Error> { | ||||||
|  |     let tz = timezone.map(|t| t.parse::<Tz>().ok()).flatten(); | ||||||
|  |     let data_opt = ContentModal::execute(ctx).await?; | ||||||
|  |  | ||||||
|  |     match data_opt { | ||||||
|  |         Some(data) => { | ||||||
|  |             create_reminder( | ||||||
|  |                 Context::Application(ctx), | ||||||
|  |                 time, | ||||||
|  |                 data.content, | ||||||
|  |                 channels, | ||||||
|  |                 interval, | ||||||
|  |                 expires, | ||||||
|  |                 tts, | ||||||
|  |                 tz, | ||||||
|  |             ) | ||||||
|  |             .await | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         None => { | ||||||
|  |             warn!("Unexpected None encountered in /multiline"); | ||||||
|  |             Ok(Context::Application(ctx) | ||||||
|  |                 .send(|m| m.content("Unexpected error.").ephemeral(true)) | ||||||
|  |                 .await | ||||||
|  |                 .map(|_| ())?) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// Create a reminder. Press "+4 more" for other options. Use "/multiline" for multiline content. | ||||||
| #[poise::command( | #[poise::command( | ||||||
|     slash_command, |     slash_command, | ||||||
|     identifying_name = "remind", |     identifying_name = "remind", | ||||||
|     default_member_permissions = "MANAGE_GUILD" |     default_member_permissions = "MANAGE_GUILD" | ||||||
| )] | )] | ||||||
| pub async fn remind( | pub async fn remind( | ||||||
|     ctx: Context<'_>, |     ctx: ApplicationContext<'_>, | ||||||
|     #[description = "A description of the time to set the reminder for"] time: String, |     #[description = "The time (and optionally date) to set the reminder for"] | ||||||
|  |     #[autocomplete = "time_hint_autocomplete"] | ||||||
|  |     time: String, | ||||||
|     #[description = "The message content to send"] content: String, |     #[description = "The message content to send"] content: String, | ||||||
|     #[description = "Channel or user mentions to set the reminder for"] channels: Option<String>, |     #[description = "Channel or user mentions to set the reminder for"] channels: Option<String>, | ||||||
|     #[description = "(Patreon only) Time to wait before repeating the reminder. Leave blank for one-shot reminder"] |     #[description = "(Patreon only) Time to wait before repeating the reminder. Leave blank for one-shot reminder"] | ||||||
|     interval: Option<String>, |     interval: Option<String>, | ||||||
|     #[description = "(Patreon only) For repeating reminders, the time at which the reminder will stop sending"] |     #[description = "(Patreon only) For repeating reminders, the time at which the reminder will stop repeating"] | ||||||
|     expires: Option<String>, |     expires: Option<String>, | ||||||
|     #[description = "Set the TTS flag on the reminder message, similar to the /tts command"] |     #[description = "Set the TTS flag on the reminder message, similar to the /tts command"] | ||||||
|     tts: Option<bool>, |     tts: Option<bool>, | ||||||
|  |     #[description = "Set a timezone override for this reminder only"] | ||||||
|  |     #[autocomplete = "timezone_autocomplete"] | ||||||
|  |     timezone: Option<String>, | ||||||
|  | ) -> Result<(), Error> { | ||||||
|  |     let tz = timezone.map(|t| t.parse::<Tz>().ok()).flatten(); | ||||||
|  |  | ||||||
|  |     create_reminder(Context::Application(ctx), time, content, channels, interval, expires, tts, tz) | ||||||
|  |         .await | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async fn create_reminder( | ||||||
|  |     ctx: Context<'_>, | ||||||
|  |     time: String, | ||||||
|  |     content: String, | ||||||
|  |     channels: Option<String>, | ||||||
|  |     interval: Option<String>, | ||||||
|  |     expires: Option<String>, | ||||||
|  |     tts: Option<bool>, | ||||||
|  |     timezone: Option<Tz>, | ||||||
| ) -> Result<(), Error> { | ) -> Result<(), Error> { | ||||||
|     if interval.is_none() && expires.is_some() { |     if interval.is_none() && expires.is_some() { | ||||||
|         ctx.say("`expires` can only be used with `interval`").await?; |         ctx.say("`expires` can only be used with `interval`").await?; | ||||||
| @@ -573,10 +658,16 @@ pub async fn remind( | |||||||
|         return Ok(()); |         return Ok(()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     ctx.defer().await?; |     let ephemeral = | ||||||
|  |         ctx.guild_data().await.map_or(false, |gr| gr.map_or(false, |g| g.ephemeral_confirmations)); | ||||||
|  |     if ephemeral { | ||||||
|  |         ctx.defer_ephemeral().await?; | ||||||
|  |     } else { | ||||||
|  |         ctx.defer().await?; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     let user_data = ctx.author_data().await.unwrap(); |     let user_data = ctx.author_data().await.unwrap(); | ||||||
|     let timezone = ctx.timezone().await; |     let timezone = timezone.unwrap_or(ctx.timezone().await); | ||||||
|  |  | ||||||
|     let time = natural_parser(&time, &timezone.to_string()).await; |     let time = natural_parser(&time, &timezone.to_string()).await; | ||||||
|  |  | ||||||
| @@ -589,8 +680,7 @@ pub async fn remind( | |||||||
|             }; |             }; | ||||||
|  |  | ||||||
|             let scopes = { |             let scopes = { | ||||||
|                 let list = |                 let list = channels.map(|arg| parse_mention_list(&arg)).unwrap_or_default(); | ||||||
|                     channels.map(|arg| parse_mention_list(&arg.to_string())).unwrap_or_default(); |  | ||||||
|  |  | ||||||
|                 if list.is_empty() { |                 if list.is_empty() { | ||||||
|                     if ctx.guild_id().is_some() { |                     if ctx.guild_id().is_some() { | ||||||
| @@ -604,13 +694,13 @@ pub async fn remind( | |||||||
|             }; |             }; | ||||||
|  |  | ||||||
|             let (processed_interval, processed_expires) = if let Some(repeat) = &interval { |             let (processed_interval, processed_expires) = if let Some(repeat) = &interval { | ||||||
|                 if check_subscription(&ctx.discord(), ctx.author().id).await |                 if check_subscription(&ctx, ctx.author().id).await | ||||||
|                     || (ctx.guild_id().is_some() |                     || (ctx.guild_id().is_some() | ||||||
|                         && check_guild_subscription(&ctx.discord(), ctx.guild_id().unwrap()).await) |                         && check_guild_subscription(&ctx, ctx.guild_id().unwrap()).await) | ||||||
|                 { |                 { | ||||||
|                     ( |                     ( | ||||||
|                         parse_duration(repeat) |                         parse_duration(repeat) | ||||||
|                             .or_else(|_| parse_duration(&format!("1 {}", repeat.to_string()))) |                             .or_else(|_| parse_duration(&format!("1 {}", repeat))) | ||||||
|                             .ok(), |                             .ok(), | ||||||
|                         { |                         { | ||||||
|                             if let Some(arg) = &expires { |                             if let Some(arg) = &expires { | ||||||
| @@ -621,9 +711,10 @@ pub async fn remind( | |||||||
|                         }, |                         }, | ||||||
|                     ) |                     ) | ||||||
|                 } else { |                 } else { | ||||||
|                     ctx.say( |                     ctx.send(|b| { | ||||||
|                         "`repeat` is only available to Patreon subscribers or self-hosted users", |                         b.content( | ||||||
|                     ) |                         "`repeat` is only available to Patreon subscribers or self-hosted users") | ||||||
|  |                     }) | ||||||
|                     .await?; |                     .await?; | ||||||
|  |  | ||||||
|                     return Ok(()); |                     return Ok(()); | ||||||
| @@ -633,13 +724,18 @@ pub async fn remind( | |||||||
|             }; |             }; | ||||||
|  |  | ||||||
|             if processed_interval.is_none() && interval.is_some() { |             if processed_interval.is_none() && interval.is_some() { | ||||||
|                 ctx.say( |                 ctx.send(|b| { | ||||||
|                     "Repeat interval could not be processed. Try similar to `1 hour` or `4 days`", |                     b.content( | ||||||
|                 ) |                     "Repeat interval could not be processed. Try similar to `1 hour` or `4 days`") | ||||||
|  |                 }) | ||||||
|                 .await?; |                 .await?; | ||||||
|             } else if processed_expires.is_none() && expires.is_some() { |             } else if processed_expires.is_none() && expires.is_some() { | ||||||
|                 ctx.say("Expiry time failed to process. Please make it as clear as possible") |                 ctx.send(|b| { | ||||||
|                     .await?; |                     b.ephemeral(true).content( | ||||||
|  |                         "Expiry time failed to process. Please make it as clear as possible", | ||||||
|  |                     ) | ||||||
|  |                 }) | ||||||
|  |                 .await?; | ||||||
|             } else { |             } else { | ||||||
|                 let mut builder = MultiReminderBuilder::new(&ctx, ctx.guild_id()) |                 let mut builder = MultiReminderBuilder::new(&ctx, ctx.guild_id()) | ||||||
|                     .author(user_data) |                     .author(user_data) | ||||||
| @@ -653,17 +749,50 @@ pub async fn remind( | |||||||
|  |  | ||||||
|                 let (errors, successes) = builder.build().await; |                 let (errors, successes) = builder.build().await; | ||||||
|  |  | ||||||
|                 let embed = create_response(successes, errors, time); |                 let embed = create_response(&successes, &errors, time); | ||||||
|  |  | ||||||
|                 ctx.send(|m| { |                 if successes.len() == 1 { | ||||||
|                     m.embed(|c| { |                     let reminder = successes.iter().next().map(|(r, _)| r.id).unwrap(); | ||||||
|                         *c = embed; |                     let undo_button = ComponentDataModel::UndoReminder(UndoReminder { | ||||||
|                         c |                         user_id: ctx.author().id, | ||||||
|  |                         reminder_id: reminder, | ||||||
|  |                     }); | ||||||
|  |  | ||||||
|  |                     ctx.send(|m| { | ||||||
|  |                         m.embed(|c| { | ||||||
|  |                             *c = embed; | ||||||
|  |                             c | ||||||
|  |                         }) | ||||||
|  |                         .components(|c| { | ||||||
|  |                             c.create_action_row(|r| { | ||||||
|  |                                 r.create_button(|b| { | ||||||
|  |                                     b.emoji(ReactionType::Unicode("🔕".to_string())) | ||||||
|  |                                         .label("Cancel") | ||||||
|  |                                         .style(ButtonStyle::Danger) | ||||||
|  |                                         .custom_id(undo_button.to_custom_id()) | ||||||
|  |                                 }) | ||||||
|  |                                 .create_button(|b| { | ||||||
|  |                                     b.emoji(ReactionType::Unicode("📝".to_string())) | ||||||
|  |                                         .label("Edit") | ||||||
|  |                                         .style(ButtonStyle::Link) | ||||||
|  |                                         .url("https://beta.reminder-bot.com/dashboard") | ||||||
|  |                                 }) | ||||||
|  |                             }) | ||||||
|  |                         }) | ||||||
|                     }) |                     }) | ||||||
|                 }) |                     .await?; | ||||||
|                 .await?; |                 } else { | ||||||
|  |                     ctx.send(|m| { | ||||||
|  |                         m.embed(|c| { | ||||||
|  |                             *c = embed; | ||||||
|  |                             c | ||||||
|  |                         }) | ||||||
|  |                     }) | ||||||
|  |                     .await?; | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         None => { |         None => { | ||||||
|             ctx.say("Time could not be processed").await?; |             ctx.say("Time could not be processed").await?; | ||||||
|         } |         } | ||||||
| @@ -673,8 +802,8 @@ pub async fn remind( | |||||||
| } | } | ||||||
|  |  | ||||||
| fn create_response( | fn create_response( | ||||||
|     successes: HashSet<ReminderScope>, |     successes: &HashSet<(Reminder, ReminderScope)>, | ||||||
|     errors: HashSet<ReminderError>, |     errors: &HashSet<ReminderError>, | ||||||
|     time: i64, |     time: i64, | ||||||
| ) -> CreateEmbed { | ) -> CreateEmbed { | ||||||
|     let success_part = match successes.len() { |     let success_part = match successes.len() { | ||||||
| @@ -682,7 +811,8 @@ fn create_response( | |||||||
|         n => format!( |         n => format!( | ||||||
|             "Reminder{s} for {locations} set for <t:{offset}:R>", |             "Reminder{s} for {locations} set for <t:{offset}:R>", | ||||||
|             s = if n > 1 { "s" } else { "" }, |             s = if n > 1 { "s" } else { "" }, | ||||||
|             locations = successes.iter().map(|l| l.mention()).collect::<Vec<String>>().join(", "), |             locations = | ||||||
|  |                 successes.iter().map(|(_, l)| l.mention()).collect::<Vec<String>>().join(", "), | ||||||
|             offset = time |             offset = time | ||||||
|         ), |         ), | ||||||
|     }; |     }; | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ use crate::{ | |||||||
|         ComponentDataModel, TodoSelector, |         ComponentDataModel, TodoSelector, | ||||||
|     }, |     }, | ||||||
|     consts::{EMBED_DESCRIPTION_MAX_LENGTH, SELECT_MAX_ENTRIES, THEME_COLOR}, |     consts::{EMBED_DESCRIPTION_MAX_LENGTH, SELECT_MAX_ENTRIES, THEME_COLOR}, | ||||||
|  |     models::CtxData, | ||||||
|     Context, Error, |     Context, Error, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @@ -116,6 +117,9 @@ pub async fn todo_channel_add( | |||||||
|     ctx: Context<'_>, |     ctx: Context<'_>, | ||||||
|     #[description = "The task to add to the todo list"] task: String, |     #[description = "The task to add to the todo list"] task: String, | ||||||
| ) -> Result<(), Error> { | ) -> Result<(), Error> { | ||||||
|  |     // ensure channel is cached | ||||||
|  |     let _ = ctx.channel_data().await; | ||||||
|  |  | ||||||
|     sqlx::query!( |     sqlx::query!( | ||||||
|         "INSERT INTO todos (guild_id, channel_id, value) |         "INSERT INTO todos (guild_id, channel_id, value) | ||||||
| VALUES ((SELECT id FROM guilds WHERE guild = ?), (SELECT id FROM channels WHERE channel = ?), ?)", | VALUES ((SELECT id FROM guilds WHERE guild = ?), (SELECT id FROM channels WHERE channel = ?), ?)", | ||||||
| @@ -336,7 +340,18 @@ pub fn show_todo_page( | |||||||
|                                 opt.create_option(|o| { |                                 opt.create_option(|o| { | ||||||
|                                     o.label(format!("Mark {} complete", count + first_num)) |                                     o.label(format!("Mark {} complete", count + first_num)) | ||||||
|                                         .value(id) |                                         .value(id) | ||||||
|                                         .description(disp.split_once(" ").unwrap_or(("", "")).1) |                                         .description({ | ||||||
|  |                                             let c = disp.split_once(' ').unwrap_or(("", "")).1; | ||||||
|  |  | ||||||
|  |                                             if c.len() > 100 { | ||||||
|  |                                                 format!( | ||||||
|  |                                                     "{}...", | ||||||
|  |                                                     c.chars().take(97).collect::<String>() | ||||||
|  |                                                 ) | ||||||
|  |                                             } else { | ||||||
|  |                                                 c.to_string() | ||||||
|  |                                             } | ||||||
|  |                                         }) | ||||||
|                                 }); |                                 }); | ||||||
|                             } |                             } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,14 +2,21 @@ pub(crate) mod pager; | |||||||
|  |  | ||||||
| use std::io::Cursor; | use std::io::Cursor; | ||||||
|  |  | ||||||
|  | use base64::{engine::general_purpose, Engine}; | ||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
| use poise::serenity::{ | use log::warn; | ||||||
|     builder::CreateEmbed, | use poise::{ | ||||||
|     client::Context, |     serenity_prelude as serenity, | ||||||
|     model::{ |     serenity_prelude::{ | ||||||
|         channel::Channel, |         builder::CreateEmbed, | ||||||
|         interactions::{message_component::MessageComponentInteraction, InteractionResponseType}, |         model::{ | ||||||
|         prelude::InteractionApplicationCommandCallbackDataFlags, |             application::interaction::{ | ||||||
|  |                 message_component::MessageComponentInteraction, InteractionResponseType, | ||||||
|  |                 MessageFlags, | ||||||
|  |             }, | ||||||
|  |             channel::Channel, | ||||||
|  |         }, | ||||||
|  |         Context, | ||||||
|     }, |     }, | ||||||
| }; | }; | ||||||
| use rmp_serde::Serializer; | use rmp_serde::Serializer; | ||||||
| @@ -17,7 +24,7 @@ use serde::{Deserialize, Serialize}; | |||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
|     commands::{ |     commands::{ | ||||||
|         moderation_cmds::{max_macro_page, show_macro_page}, |         command_macro::list::{max_macro_page, show_macro_page}, | ||||||
|         reminder_cmds::{max_delete_page, show_delete_page}, |         reminder_cmds::{max_delete_page, show_delete_page}, | ||||||
|         todo_cmds::{max_todo_page, show_todo_page}, |         todo_cmds::{max_todo_page, show_todo_page}, | ||||||
|     }, |     }, | ||||||
| @@ -38,17 +45,19 @@ pub enum ComponentDataModel { | |||||||
|     DelSelector(DelSelector), |     DelSelector(DelSelector), | ||||||
|     TodoSelector(TodoSelector), |     TodoSelector(TodoSelector), | ||||||
|     MacroPager(MacroPager), |     MacroPager(MacroPager), | ||||||
|  |     UndoReminder(UndoReminder), | ||||||
| } | } | ||||||
|  |  | ||||||
| impl ComponentDataModel { | impl ComponentDataModel { | ||||||
|     pub fn to_custom_id(&self) -> String { |     pub fn to_custom_id(&self) -> String { | ||||||
|         let mut buf = Vec::new(); |         let mut buf = Vec::new(); | ||||||
|         self.serialize(&mut Serializer::new(&mut buf)).unwrap(); |         self.serialize(&mut Serializer::new(&mut buf)).unwrap(); | ||||||
|         base64::encode(buf) |         general_purpose::STANDARD.encode(buf) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn from_custom_id(data: &String) -> Self { |     pub fn from_custom_id(data: &String) -> Self { | ||||||
|         let buf = base64::decode(data) |         let buf = general_purpose::STANDARD | ||||||
|  |             .decode(data) | ||||||
|             .map_err(|e| format!("Could not decode `custom_id' {}: {:?}", data, e)) |             .map_err(|e| format!("Could not decode `custom_id' {}: {:?}", data, e)) | ||||||
|             .unwrap(); |             .unwrap(); | ||||||
|         let cur = Cursor::new(buf); |         let cur = Cursor::new(buf); | ||||||
| @@ -106,7 +115,7 @@ impl ComponentDataModel { | |||||||
|                         char_count < EMBED_DESCRIPTION_MAX_LENGTH |                         char_count < EMBED_DESCRIPTION_MAX_LENGTH | ||||||
|                     }) |                     }) | ||||||
|                     .collect::<Vec<String>>() |                     .collect::<Vec<String>>() | ||||||
|                     .join("\n"); |                     .join(""); | ||||||
|  |  | ||||||
|                 let mut embed = CreateEmbed::default(); |                 let mut embed = CreateEmbed::default(); | ||||||
|                 embed |                 embed | ||||||
| @@ -159,10 +168,13 @@ impl ComponentDataModel { | |||||||
|             ComponentDataModel::DelSelector(selector) => { |             ComponentDataModel::DelSelector(selector) => { | ||||||
|                 let selected_id = component.data.values.join(","); |                 let selected_id = component.data.values.join(","); | ||||||
|  |  | ||||||
|                 sqlx::query!("DELETE FROM reminders WHERE FIND_IN_SET(id, ?)", selected_id) |                 sqlx::query!( | ||||||
|                     .execute(&data.database) |                     "UPDATE reminders SET `status` = 'deleted' WHERE FIND_IN_SET(id, ?)", | ||||||
|                     .await |                     selected_id | ||||||
|                     .unwrap(); |                 ) | ||||||
|  |                 .execute(&data.database) | ||||||
|  |                 .await | ||||||
|  |                 .unwrap(); | ||||||
|  |  | ||||||
|                 let reminders = Reminder::from_guild( |                 let reminders = Reminder::from_guild( | ||||||
|                     &ctx, |                     &ctx, | ||||||
| @@ -253,7 +265,7 @@ WHERE guilds.guild = ?", | |||||||
|                             r.kind(InteractionResponseType::ChannelMessageWithSource) |                             r.kind(InteractionResponseType::ChannelMessageWithSource) | ||||||
|                                 .interaction_response_data(|d| { |                                 .interaction_response_data(|d| { | ||||||
|                                     d.flags( |                                     d.flags( | ||||||
|                                         InteractionApplicationCommandCallbackDataFlags::EPHEMERAL, |                                         MessageFlags::EPHEMERAL, | ||||||
|                                     ) |                                     ) | ||||||
|                                     .content("Only the user who performed the command can use these components") |                                     .content("Only the user who performed the command can use these components") | ||||||
|                                 }) |                                 }) | ||||||
| @@ -307,7 +319,7 @@ WHERE guilds.guild = ?", | |||||||
|                             r.kind(InteractionResponseType::ChannelMessageWithSource) |                             r.kind(InteractionResponseType::ChannelMessageWithSource) | ||||||
|                                 .interaction_response_data(|d| { |                                 .interaction_response_data(|d| { | ||||||
|                                     d.flags( |                                     d.flags( | ||||||
|                                         InteractionApplicationCommandCallbackDataFlags::EPHEMERAL, |                                         MessageFlags::EPHEMERAL, | ||||||
|                                     ) |                                     ) | ||||||
|                                     .content("Only the user who performed the command can use these components") |                                     .content("Only the user who performed the command can use these components") | ||||||
|                                 }) |                                 }) | ||||||
| @@ -334,6 +346,70 @@ WHERE guilds.guild = ?", | |||||||
|                     }) |                     }) | ||||||
|                     .await; |                     .await; | ||||||
|             } |             } | ||||||
|  |             ComponentDataModel::UndoReminder(undo_reminder) => { | ||||||
|  |                 if component.user.id == undo_reminder.user_id { | ||||||
|  |                     let reminder = | ||||||
|  |                         Reminder::from_id(&data.database, undo_reminder.reminder_id).await; | ||||||
|  |  | ||||||
|  |                     if let Some(reminder) = reminder { | ||||||
|  |                         match reminder.delete(&data.database).await { | ||||||
|  |                             Ok(()) => { | ||||||
|  |                                 let _ = component | ||||||
|  |                                     .create_interaction_response(&ctx, |f| { | ||||||
|  |                                         f.kind(InteractionResponseType::UpdateMessage) | ||||||
|  |                                             .interaction_response_data(|d| { | ||||||
|  |                                                 d.embed(|e| { | ||||||
|  |                                                     e.title("Reminder Canceled") | ||||||
|  |                                                         .description( | ||||||
|  |                                                             "This reminder has been canceled.", | ||||||
|  |                                                         ) | ||||||
|  |                                                         .color(*THEME_COLOR) | ||||||
|  |                                                 }) | ||||||
|  |                                                 .components(|c| c) | ||||||
|  |                                             }) | ||||||
|  |                                     }) | ||||||
|  |                                     .await; | ||||||
|  |                             } | ||||||
|  |                             Err(e) => { | ||||||
|  |                                 warn!("Error canceling reminder: {:?}", e); | ||||||
|  |  | ||||||
|  |                                 let _ = component | ||||||
|  |                                     .create_interaction_response(&ctx, |f| { | ||||||
|  |                                         f.kind(InteractionResponseType::ChannelMessageWithSource) | ||||||
|  |                                             .interaction_response_data(|d| { | ||||||
|  |                                                 d.content( | ||||||
|  |                                                     "The reminder could not be canceled: it may have already been deleted. Check `/del`!") | ||||||
|  |                                                     .ephemeral(true) | ||||||
|  |                                             }) | ||||||
|  |                                     }) | ||||||
|  |                                     .await; | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     } else { | ||||||
|  |                         let _ = component | ||||||
|  |                             .create_interaction_response(&ctx, |f| { | ||||||
|  |                                 f.kind(InteractionResponseType::ChannelMessageWithSource) | ||||||
|  |                                     .interaction_response_data(|d| { | ||||||
|  |                                         d.content( | ||||||
|  |                                             "The reminder could not be canceled: it may have already been deleted. Check `/del`!") | ||||||
|  |                                             .ephemeral(true) | ||||||
|  |                                     }) | ||||||
|  |                             }) | ||||||
|  |                             .await; | ||||||
|  |                     } | ||||||
|  |                 } else { | ||||||
|  |                     let _ = component | ||||||
|  |                         .create_interaction_response(&ctx, |f| { | ||||||
|  |                             f.kind(InteractionResponseType::ChannelMessageWithSource) | ||||||
|  |                                 .interaction_response_data(|d| { | ||||||
|  |                                     d.content( | ||||||
|  |                                         "Only the user who performed the command can use this button.") | ||||||
|  |                                         .ephemeral(true) | ||||||
|  |                                 }) | ||||||
|  |                         }) | ||||||
|  |                         .await; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -351,3 +427,9 @@ pub struct TodoSelector { | |||||||
|     pub channel_id: Option<u64>, |     pub channel_id: Option<u64>, | ||||||
|     pub guild_id: Option<u64>, |     pub guild_id: Option<u64>, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | #[derive(Serialize, Deserialize)] | ||||||
|  | pub struct UndoReminder { | ||||||
|  |     pub user_id: serenity::UserId, | ||||||
|  |     pub reminder_id: u32, | ||||||
|  | } | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| // todo split pager out into a single struct | // todo split pager out into a single struct | ||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
| use poise::serenity::{ | use poise::serenity_prelude::{ | ||||||
|     builder::CreateComponents, model::interactions::message_component::ButtonStyle, |     builder::CreateComponents, model::application::component::ButtonStyle, | ||||||
| }; | }; | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
| use serde_repr::*; | use serde_repr::*; | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ pub const DAY: u64 = 86_400; | |||||||
| pub const HOUR: u64 = 3_600; | pub const HOUR: u64 = 3_600; | ||||||
| pub const MINUTE: u64 = 60; | pub const MINUTE: u64 = 60; | ||||||
|  |  | ||||||
| pub const EMBED_DESCRIPTION_MAX_LENGTH: usize = 4000; | pub const EMBED_DESCRIPTION_MAX_LENGTH: usize = 4096; | ||||||
| pub const SELECT_MAX_ENTRIES: usize = 25; | pub const SELECT_MAX_ENTRIES: usize = 25; | ||||||
|  |  | ||||||
| pub const CHARACTERS: &str = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_"; | pub const CHARACTERS: &str = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_"; | ||||||
| @@ -12,22 +12,18 @@ pub const MACRO_MAX_COMMANDS: usize = 5; | |||||||
|  |  | ||||||
| use std::{collections::HashSet, env, iter::FromIterator}; | use std::{collections::HashSet, env, iter::FromIterator}; | ||||||
|  |  | ||||||
| use poise::serenity::model::prelude::AttachmentType; | use poise::serenity_prelude::model::prelude::AttachmentType; | ||||||
| use regex::Regex; | use regex::Regex; | ||||||
|  |  | ||||||
| lazy_static! { | lazy_static! { | ||||||
|     pub static ref DEFAULT_AVATAR: AttachmentType<'static> = ( |     pub static ref DEFAULT_AVATAR: AttachmentType<'static> = ( | ||||||
|         include_bytes!(concat!( |         include_bytes!(concat!(env!("CARGO_MANIFEST_DIR"), "/assets/webhook.jpg")) as &[u8], | ||||||
|             env!("CARGO_MANIFEST_DIR"), |         "webhook.jpg", | ||||||
|             "/assets/", |  | ||||||
|             env!("WEBHOOK_AVATAR", "WEBHOOK_AVATAR not provided for compilation") |  | ||||||
|         )) as &[u8], |  | ||||||
|         env!("WEBHOOK_AVATAR"), |  | ||||||
|     ) |     ) | ||||||
|         .into(); |         .into(); | ||||||
|     pub static ref REGEX_CHANNEL_USER: Regex = Regex::new(r#"\s*<(#|@)(?:!)?(\d+)>\s*"#).unwrap(); |     pub static ref REGEX_CHANNEL_USER: Regex = Regex::new(r#"\s*<(#|@)(?:!)?(\d+)>\s*"#).unwrap(); | ||||||
|     pub static ref SUBSCRIPTION_ROLES: HashSet<u64> = HashSet::from_iter( |     pub static ref SUBSCRIPTION_ROLES: HashSet<u64> = HashSet::from_iter( | ||||||
|         env::var("SUBSCRIPTION_ROLES") |         env::var("PATREON_ROLE_ID") | ||||||
|             .map(|var| var |             .map(|var| var | ||||||
|                 .split(',') |                 .split(',') | ||||||
|                 .filter_map(|item| { item.parse::<u64>().ok() }) |                 .filter_map(|item| { item.parse::<u64>().ok() }) | ||||||
| @@ -35,16 +31,12 @@ lazy_static! { | |||||||
|             .unwrap_or_else(|_| Vec::new()) |             .unwrap_or_else(|_| Vec::new()) | ||||||
|     ); |     ); | ||||||
|     pub static ref CNC_GUILD: Option<u64> = |     pub static ref CNC_GUILD: Option<u64> = | ||||||
|         env::var("CNC_GUILD").map(|var| var.parse::<u64>().ok()).ok().flatten(); |         env::var("PATREON_GUILD_ID").map(|var| var.parse::<u64>().ok()).ok().flatten(); | ||||||
|     pub static ref MIN_INTERVAL: i64 = env::var("MIN_INTERVAL") |     pub static ref MIN_INTERVAL: i64 = | ||||||
|         .ok() |         env::var("MIN_INTERVAL").ok().and_then(|inner| inner.parse::<i64>().ok()).unwrap_or(600); | ||||||
|         .map(|inner| inner.parse::<i64>().ok()) |  | ||||||
|         .flatten() |  | ||||||
|         .unwrap_or(600); |  | ||||||
|     pub static ref MAX_TIME: i64 = env::var("MAX_TIME") |     pub static ref MAX_TIME: i64 = env::var("MAX_TIME") | ||||||
|         .ok() |         .ok() | ||||||
|         .map(|inner| inner.parse::<i64>().ok()) |         .and_then(|inner| inner.parse::<i64>().ok()) | ||||||
|         .flatten() |  | ||||||
|         .unwrap_or(60 * 60 * 24 * 365 * 50); |         .unwrap_or(60 * 60 * 24 * 365 * 50); | ||||||
|     pub static ref LOCAL_TIMEZONE: String = |     pub static ref LOCAL_TIMEZONE: String = | ||||||
|         env::var("LOCAL_TIMEZONE").unwrap_or_else(|_| "UTC".to_string()); |         env::var("LOCAL_TIMEZONE").unwrap_or_else(|_| "UTC".to_string()); | ||||||
| @@ -52,5 +44,5 @@ lazy_static! { | |||||||
|         .map_or(THEME_COLOR_FALLBACK, |inner| u32::from_str_radix(&inner, 16) |         .map_or(THEME_COLOR_FALLBACK, |inner| u32::from_str_radix(&inner, 16) | ||||||
|             .unwrap_or(THEME_COLOR_FALLBACK)); |             .unwrap_or(THEME_COLOR_FALLBACK)); | ||||||
|     pub static ref PYTHON_LOCATION: String = |     pub static ref PYTHON_LOCATION: String = | ||||||
|         env::var("PYTHON_LOCATION").unwrap_or_else(|_| "venv/bin/python3".to_string()); |         env::var("PYTHON_LOCATION").unwrap_or_else(|_| "/usr/bin/python3".to_string()); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,12 +1,12 @@ | |||||||
| use std::{collections::HashMap, env, sync::atomic::Ordering}; | use std::{collections::HashMap, env}; | ||||||
|  |  | ||||||
| use log::{error, info, warn}; | use log::error; | ||||||
| use poise::{ | use poise::{ | ||||||
|     serenity::{model::interactions::Interaction, utils::shard_id}, |  | ||||||
|     serenity_prelude as serenity, |     serenity_prelude as serenity, | ||||||
|  |     serenity_prelude::{model::application::interaction::Interaction, utils::shard_id}, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| use crate::{component_models::ComponentDataModel, Data, Error}; | use crate::{component_models::ComponentDataModel, Data, Error, THEME_COLOR}; | ||||||
|  |  | ||||||
| pub async fn listener( | pub async fn listener( | ||||||
|     ctx: &serenity::Context, |     ctx: &serenity::Context, | ||||||
| @@ -14,44 +14,8 @@ pub async fn listener( | |||||||
|     data: &Data, |     data: &Data, | ||||||
| ) -> Result<(), Error> { | ) -> Result<(), Error> { | ||||||
|     match event { |     match event { | ||||||
|         poise::Event::CacheReady { .. } => { |         poise::Event::Ready { .. } => { | ||||||
|             info!("Cache Ready! Preparing extra processes"); |             ctx.set_activity(serenity::Activity::watching("for /remind")).await; | ||||||
|  |  | ||||||
|             if !data.is_loop_running.load(Ordering::Relaxed) { |  | ||||||
|                 let kill_tx = data.broadcast.clone(); |  | ||||||
|                 let kill_recv = data.broadcast.subscribe(); |  | ||||||
|  |  | ||||||
|                 let ctx1 = ctx.clone(); |  | ||||||
|                 let ctx2 = ctx.clone(); |  | ||||||
|  |  | ||||||
|                 let pool1 = data.database.clone(); |  | ||||||
|                 let pool2 = data.database.clone(); |  | ||||||
|  |  | ||||||
|                 let run_settings = env::var("DONTRUN").unwrap_or_else(|_| "".to_string()); |  | ||||||
|  |  | ||||||
|                 if !run_settings.contains("postman") { |  | ||||||
|                     tokio::spawn(async move { |  | ||||||
|                         match postman::initialize(kill_recv, ctx1, &pool1).await { |  | ||||||
|                             Ok(_) => {} |  | ||||||
|                             Err(e) => { |  | ||||||
|                                 error!("postman exiting: {}", e); |  | ||||||
|                             } |  | ||||||
|                         }; |  | ||||||
|                     }); |  | ||||||
|                 } else { |  | ||||||
|                     warn!("Not running postman") |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 if !run_settings.contains("web") { |  | ||||||
|                     tokio::spawn(async move { |  | ||||||
|                         reminder_web::initialize(kill_tx, ctx2, pool2).await.unwrap(); |  | ||||||
|                     }); |  | ||||||
|                 } else { |  | ||||||
|                     warn!("Not running web") |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 data.is_loop_running.swap(true, Ordering::Relaxed); |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|         poise::Event::ChannelDelete { channel } => { |         poise::Event::ChannelDelete { channel } => { | ||||||
|             sqlx::query!("DELETE FROM channels WHERE channel = ?", channel.id.as_u64()) |             sqlx::query!("DELETE FROM channels WHERE channel = ?", channel.id.as_u64()) | ||||||
| @@ -63,46 +27,36 @@ pub async fn listener( | |||||||
|             if *is_new { |             if *is_new { | ||||||
|                 let guild_id = guild.id.as_u64().to_owned(); |                 let guild_id = guild.id.as_u64().to_owned(); | ||||||
|  |  | ||||||
|                 sqlx::query!("INSERT INTO guilds (guild) VALUES (?)", guild_id) |                 sqlx::query!("INSERT IGNORE INTO guilds (guild) VALUES (?)", guild_id) | ||||||
|                     .execute(&data.database) |                     .execute(&data.database) | ||||||
|                     .await |                     .await?; | ||||||
|                     .unwrap(); |  | ||||||
|  |  | ||||||
|                 if let Ok(token) = env::var("DISCORDBOTS_TOKEN") { |                 if let Err(e) = post_guild_count(ctx, &data.http, guild_id).await { | ||||||
|                     let shard_count = ctx.cache.shard_count(); |                     error!("DiscordBotList: {:?}", e); | ||||||
|                     let current_shard_id = shard_id(guild_id, shard_count); |                 } | ||||||
|  |  | ||||||
|                     let guild_count = ctx |                 let default_channel = guild.default_channel_guaranteed(); | ||||||
|                         .cache |  | ||||||
|                         .guilds() |                 if let Some(default_channel) = default_channel { | ||||||
|                         .iter() |                     default_channel | ||||||
|                         .filter(|g| { |                         .send_message(&ctx, |m| { | ||||||
|                             shard_id(g.as_u64().to_owned(), shard_count) == current_shard_id |                             m.embed(|e| { | ||||||
|  |                                 e.title("Thank you for adding Reminder Bot!").description( | ||||||
|  |                                     "To get started: | ||||||
|  | • Set your timezone with `/timezone` | ||||||
|  | • Set up permissions in Server Settings 🠚 Integrations 🠚 Reminder Bot (desktop only) | ||||||
|  | • Create your first reminder with `/remind` | ||||||
|  |  | ||||||
|  | __Support__ | ||||||
|  | If you need any support, please come and ask us! Join our [Discord](https://discord.jellywx.com). | ||||||
|  |  | ||||||
|  | __Updates__ | ||||||
|  | To stay up to date on the latest features and fixes, join our [Discord](https://discord.jellywx.com). | ||||||
|  | ", | ||||||
|  |                                 ).color(*THEME_COLOR) | ||||||
|  |                             }) | ||||||
|                         }) |                         }) | ||||||
|                         .count() as u64; |                         .await?; | ||||||
|  |  | ||||||
|                     let mut hm = HashMap::new(); |  | ||||||
|                     hm.insert("server_count", guild_count); |  | ||||||
|                     hm.insert("shard_id", current_shard_id); |  | ||||||
|                     hm.insert("shard_count", shard_count); |  | ||||||
|  |  | ||||||
|                     let response = data |  | ||||||
|                         .http |  | ||||||
|                         .post( |  | ||||||
|                             format!( |  | ||||||
|                                 "https://top.gg/api/bots/{}/stats", |  | ||||||
|                                 ctx.cache.current_user_id().as_u64() |  | ||||||
|                             ) |  | ||||||
|                             .as_str(), |  | ||||||
|                         ) |  | ||||||
|                         .header("Authorization", token) |  | ||||||
|                         .json(&hm) |  | ||||||
|                         .send() |  | ||||||
|                         .await; |  | ||||||
|  |  | ||||||
|                     if let Err(res) = response { |  | ||||||
|                         println!("DiscordBots Response: {:?}", res); |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -111,16 +65,50 @@ pub async fn listener( | |||||||
|                 .execute(&data.database) |                 .execute(&data.database) | ||||||
|                 .await; |                 .await; | ||||||
|         } |         } | ||||||
|         poise::Event::InteractionCreate { interaction } => match interaction { |         poise::Event::InteractionCreate { interaction } => { | ||||||
|             Interaction::MessageComponent(component) => { |             if let Interaction::MessageComponent(component) = interaction { | ||||||
|                 let component_model = ComponentDataModel::from_custom_id(&component.data.custom_id); |                 let component_model = ComponentDataModel::from_custom_id(&component.data.custom_id); | ||||||
|  |  | ||||||
|                 component_model.act(ctx, data, component).await; |                 component_model.act(ctx, data, component).await; | ||||||
|             } |             } | ||||||
|             _ => {} |         } | ||||||
|         }, |  | ||||||
|         _ => {} |         _ => {} | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | async fn post_guild_count( | ||||||
|  |     ctx: &serenity::Context, | ||||||
|  |     http: &reqwest::Client, | ||||||
|  |     guild_id: u64, | ||||||
|  | ) -> Result<(), reqwest::Error> { | ||||||
|  |     if let Ok(token) = env::var("DISCORDBOTS_TOKEN") { | ||||||
|  |         let shard_count = ctx.cache.shard_count(); | ||||||
|  |         let current_shard_id = shard_id(guild_id, shard_count); | ||||||
|  |  | ||||||
|  |         let guild_count = ctx | ||||||
|  |             .cache | ||||||
|  |             .guilds() | ||||||
|  |             .iter() | ||||||
|  |             .filter(|g| shard_id(g.as_u64().to_owned(), shard_count) == current_shard_id) | ||||||
|  |             .count() as u64; | ||||||
|  |  | ||||||
|  |         let mut hm = HashMap::new(); | ||||||
|  |         hm.insert("server_count", guild_count); | ||||||
|  |         hm.insert("shard_id", current_shard_id); | ||||||
|  |         hm.insert("shard_count", shard_count); | ||||||
|  |  | ||||||
|  |         http.post( | ||||||
|  |             format!("https://top.gg/api/bots/{}/stats", ctx.cache.current_user_id().as_u64()) | ||||||
|  |                 .as_str(), | ||||||
|  |         ) | ||||||
|  |         .header("Authorization", token) | ||||||
|  |         .json(&hm) | ||||||
|  |         .send() | ||||||
|  |         .await | ||||||
|  |         .map(|_| ()) | ||||||
|  |     } else { | ||||||
|  |         Ok(()) | ||||||
|  |     } | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										88
									
								
								src/hooks.rs
									
									
									
									
									
								
							
							
						
						| @@ -1,72 +1,72 @@ | |||||||
| use poise::serenity::model::channel::Channel; | use poise::{ | ||||||
|  |     serenity_prelude::model::channel::Channel, ApplicationCommandOrAutocompleteInteraction, | ||||||
|  | }; | ||||||
|  |  | ||||||
| use crate::{consts::MACRO_MAX_COMMANDS, models::command_macro::RecordedCommand, Context, Error}; | use crate::{consts::MACRO_MAX_COMMANDS, models::command_macro::RecordedCommand, Context, Error}; | ||||||
|  |  | ||||||
| async fn macro_check(ctx: Context<'_>) -> bool { | async fn macro_check(ctx: Context<'_>) -> bool { | ||||||
|     if let Context::Application(app_ctx) = ctx { |     if let Context::Application(app_ctx) = ctx { | ||||||
|         if let Some(guild_id) = ctx.guild_id() { |         if let ApplicationCommandOrAutocompleteInteraction::ApplicationCommand(_) = | ||||||
|             if ctx.command().identifying_name != "finish_macro" { |             app_ctx.interaction | ||||||
|                 let mut lock = ctx.data().recording_macros.write().await; |         { | ||||||
|  |             if let Some(guild_id) = ctx.guild_id() { | ||||||
|  |                 if ctx.command().identifying_name != "finish_macro" { | ||||||
|  |                     let mut lock = ctx.data().recording_macros.write().await; | ||||||
|  |  | ||||||
|                 if let Some(command_macro) = lock.get_mut(&(guild_id, ctx.author().id)) { |                     if let Some(command_macro) = lock.get_mut(&(guild_id, ctx.author().id)) { | ||||||
|                     if command_macro.commands.len() >= MACRO_MAX_COMMANDS { |                         if command_macro.commands.len() >= MACRO_MAX_COMMANDS { | ||||||
|                         let _ = ctx.send(|m| { |                             let _ = ctx.send(|m| { | ||||||
|                                 m.ephemeral(true).content( |                             m.ephemeral(true).content( | ||||||
|                                     format!("{} commands already recorded. Please use `/macro finish` to end recording.", MACRO_MAX_COMMANDS), |                                 format!("{} commands already recorded. Please use `/macro finish` to end recording.", MACRO_MAX_COMMANDS), | ||||||
|                                 ) |                             ) | ||||||
|                             }) |                         }) | ||||||
|                             .await; |                             .await; | ||||||
|                     } else { |                         } else { | ||||||
|                         let recorded = RecordedCommand { |                             let recorded = RecordedCommand { | ||||||
|                             action: None, |                                 action: None, | ||||||
|                             command_name: ctx.command().identifying_name.clone(), |                                 command_name: ctx.command().identifying_name.clone(), | ||||||
|                             options: Vec::from(app_ctx.args), |                                 options: Vec::from(app_ctx.args), | ||||||
|                         }; |                             }; | ||||||
|  |  | ||||||
|                         command_macro.commands.push(recorded); |                             command_macro.commands.push(recorded); | ||||||
|  |  | ||||||
|                         let _ = ctx |                             let _ = ctx | ||||||
|                             .send(|m| m.ephemeral(true).content("Command recorded to macro")) |                                 .send(|m| m.ephemeral(true).content("Command recorded to macro")) | ||||||
|                             .await; |                                 .await; | ||||||
|  |                         } | ||||||
|  |  | ||||||
|  |                         return false; | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|                     false |  | ||||||
|                 } else { |  | ||||||
|                     true |  | ||||||
|                 } |                 } | ||||||
|             } else { |  | ||||||
|                 true |  | ||||||
|             } |             } | ||||||
|         } else { |  | ||||||
|             true |  | ||||||
|         } |         } | ||||||
|     } else { |  | ||||||
|         true |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     true | ||||||
| } | } | ||||||
|  |  | ||||||
| async fn check_self_permissions(ctx: Context<'_>) -> bool { | async fn check_self_permissions(ctx: Context<'_>) -> bool { | ||||||
|     if let Some(guild) = ctx.guild() { |     if let Some(guild) = ctx.guild() { | ||||||
|         let user_id = ctx.discord().cache.current_user_id(); |         let user_id = ctx.serenity_context().cache.current_user_id(); | ||||||
|  |  | ||||||
|  |         let manage_webhooks = | ||||||
|  |             guild.member_permissions(&ctx, user_id).await.map_or(false, |p| p.manage_webhooks()); | ||||||
|  |  | ||||||
|         let manage_webhooks = guild |  | ||||||
|             .member_permissions(&ctx.discord(), user_id) |  | ||||||
|             .await |  | ||||||
|             .map_or(false, |p| p.manage_webhooks()); |  | ||||||
|         let (view_channel, send_messages, embed_links) = ctx |         let (view_channel, send_messages, embed_links) = ctx | ||||||
|             .channel_id() |             .channel_id() | ||||||
|             .to_channel_cached(&ctx.discord()) |             .to_channel(&ctx) | ||||||
|             .map(|c| { |             .await | ||||||
|  |             .ok() | ||||||
|  |             .and_then(|c| { | ||||||
|                 if let Channel::Guild(channel) = c { |                 if let Channel::Guild(channel) = c { | ||||||
|                     channel.permissions_for_user(&ctx.discord(), user_id).ok() |                     let perms = channel.permissions_for_user(&ctx, user_id).ok()?; | ||||||
|  |  | ||||||
|  |                     Some((perms.view_channel(), perms.send_messages(), perms.embed_links())) | ||||||
|                 } else { |                 } else { | ||||||
|                     None |                     None | ||||||
|                 } |                 } | ||||||
|             }) |             }) | ||||||
|             .flatten() |             .unwrap_or((false, false, false)); | ||||||
|             .map_or((false, false, false), |p| { |  | ||||||
|                 (p.view_channel(), p.send_messages(), p.embed_links()) |  | ||||||
|             }); |  | ||||||
|  |  | ||||||
|         if manage_webhooks && send_messages && embed_links { |         if manage_webhooks && send_messages && embed_links { | ||||||
|             true |             true | ||||||
| @@ -82,8 +82,8 @@ async fn check_self_permissions(ctx: Context<'_>) -> bool { | |||||||
| {}     **Manage Webhooks**", | {}     **Manage Webhooks**", | ||||||
|                         if view_channel { "✅" } else { "❌" }, |                         if view_channel { "✅" } else { "❌" }, | ||||||
|                         if send_messages { "✅" } else { "❌" }, |                         if send_messages { "✅" } else { "❌" }, | ||||||
|                         if manage_webhooks { "✅" } else { "❌" }, |  | ||||||
|                         if embed_links { "✅" } else { "❌" }, |                         if embed_links { "✅" } else { "❌" }, | ||||||
|  |                         if manage_webhooks { "✅" } else { "❌" }, | ||||||
|                     )) |                     )) | ||||||
|                 }) |                 }) | ||||||
|                 .await; |                 .await; | ||||||
|   | |||||||
| @@ -75,7 +75,7 @@ impl fmt::Display for Error { | |||||||
|         match self { |         match self { | ||||||
|             Error::InvalidCharacter(offset) => write!(f, "invalid character at {}", offset), |             Error::InvalidCharacter(offset) => write!(f, "invalid character at {}", offset), | ||||||
|             Error::NumberExpected(offset) => write!(f, "expected number at {}", offset), |             Error::NumberExpected(offset) => write!(f, "expected number at {}", offset), | ||||||
|             Error::UnknownUnit { unit, value, .. } if &unit == &"" => { |             Error::UnknownUnit { unit, value, .. } if unit.is_empty() => { | ||||||
|                 write!(f, "time unit needed, for example {0}sec or {0}ms", value,) |                 write!(f, "time unit needed, for example {0}sec or {0}ms", value,) | ||||||
|             } |             } | ||||||
|             Error::UnknownUnit { unit, .. } => { |             Error::UnknownUnit { unit, .. } => { | ||||||
| @@ -110,13 +110,14 @@ impl OverflowOp for u64 { | |||||||
| #[derive(Copy, Clone)] | #[derive(Copy, Clone)] | ||||||
| pub struct Interval { | pub struct Interval { | ||||||
|     pub month: u64, |     pub month: u64, | ||||||
|  |     pub day: u64, | ||||||
|     pub sec: u64, |     pub sec: u64, | ||||||
| } | } | ||||||
|  |  | ||||||
| struct Parser<'a> { | struct Parser<'a> { | ||||||
|     iter: Chars<'a>, |     iter: Chars<'a>, | ||||||
|     src: &'a str, |     src: &'a str, | ||||||
|     current: (u64, u64, u64), |     current: (u64, u64, u64, u64), | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'a> Parser<'a> { | impl<'a> Parser<'a> { | ||||||
| @@ -140,17 +141,17 @@ impl<'a> Parser<'a> { | |||||||
|         Ok(None) |         Ok(None) | ||||||
|     } |     } | ||||||
|     fn parse_unit(&mut self, n: u64, start: usize, end: usize) -> Result<(), Error> { |     fn parse_unit(&mut self, n: u64, start: usize, end: usize) -> Result<(), Error> { | ||||||
|         let (mut month, mut sec, nsec) = match &self.src[start..end] { |         let (mut month, mut day, mut sec, nsec) = match &self.src[start..end] { | ||||||
|             "nanos" | "nsec" | "ns" => (0u64, 0u64, n), |             "nanos" | "nsec" | "ns" => (0, 0u64, 0u64, n), | ||||||
|             "usec" | "us" => (0, 0u64, n.mul(1000)?), |             "usec" | "us" => (0, 0, 0u64, n.mul(1000)?), | ||||||
|             "millis" | "msec" | "ms" => (0, 0u64, n.mul(1_000_000)?), |             "millis" | "msec" | "ms" => (0, 0, 0u64, n.mul(1_000_000)?), | ||||||
|             "seconds" | "second" | "secs" | "sec" | "s" => (0, n, 0), |             "seconds" | "second" | "secs" | "sec" | "s" => (0, 0, n, 0), | ||||||
|             "minutes" | "minute" | "min" | "mins" | "m" => (0, n.mul(60)?, 0), |             "minutes" | "minute" | "min" | "mins" | "m" => (0, 0, n.mul(60)?, 0), | ||||||
|             "hours" | "hour" | "hr" | "hrs" | "h" => (0, n.mul(3600)?, 0), |             "hours" | "hour" | "hr" | "hrs" | "h" => (0, 0, n.mul(3600)?, 0), | ||||||
|             "days" | "day" | "d" => (0, n.mul(86400)?, 0), |             "days" | "day" | "d" => (0, n, 0, 0), | ||||||
|             "weeks" | "week" | "w" => (0, n.mul(86400 * 7)?, 0), |             "weeks" | "week" | "w" => (0, n.mul(7)?, 0, 0), | ||||||
|             "months" | "month" | "M" => (n, 0, 0), |             "months" | "month" => (n, 0, 0, 0), | ||||||
|             "years" | "year" | "y" => (12, 0, 0), |             "years" | "year" | "y" => (n.mul(12)?, 0, 0, 0), | ||||||
|             _ => { |             _ => { | ||||||
|                 return Err(Error::UnknownUnit { |                 return Err(Error::UnknownUnit { | ||||||
|                     start, |                     start, | ||||||
| @@ -160,15 +161,16 @@ impl<'a> Parser<'a> { | |||||||
|                 }); |                 }); | ||||||
|             } |             } | ||||||
|         }; |         }; | ||||||
|         let mut nsec = self.current.2 + nsec; |         let mut nsec = self.current.3 + nsec; | ||||||
|         if nsec > 1_000_000_000 { |         if nsec > 1_000_000_000 { | ||||||
|             sec = sec + nsec / 1_000_000_000; |             sec += nsec / 1_000_000_000; | ||||||
|             nsec %= 1_000_000_000; |             nsec %= 1_000_000_000; | ||||||
|         } |         } | ||||||
|         sec = self.current.1 + sec; |         sec += self.current.2; | ||||||
|         month = self.current.0 + month; |         day += self.current.1; | ||||||
|  |         month += self.current.0; | ||||||
|  |  | ||||||
|         self.current = (month, sec, nsec); |         self.current = (month, day, sec, nsec); | ||||||
|  |  | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
| @@ -215,7 +217,13 @@ impl<'a> Parser<'a> { | |||||||
|             self.parse_unit(n, start, off)?; |             self.parse_unit(n, start, off)?; | ||||||
|             n = match self.parse_first_char()? { |             n = match self.parse_first_char()? { | ||||||
|                 Some(n) => n, |                 Some(n) => n, | ||||||
|                 None => return Ok(Interval { month: self.current.0, sec: self.current.1 }), |                 None => { | ||||||
|  |                     return Ok(Interval { | ||||||
|  |                         month: self.current.0, | ||||||
|  |                         day: self.current.1, | ||||||
|  |                         sec: self.current.2, | ||||||
|  |                     }) | ||||||
|  |                 } | ||||||
|             }; |             }; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -247,5 +255,82 @@ impl<'a> Parser<'a> { | |||||||
| /// assert_eq!(parse_duration("32ms"), Ok(Duration::new(0, 32_000_000))); | /// assert_eq!(parse_duration("32ms"), Ok(Duration::new(0, 32_000_000))); | ||||||
| /// ``` | /// ``` | ||||||
| pub fn parse_duration(s: &str) -> Result<Interval, Error> { | pub fn parse_duration(s: &str) -> Result<Interval, Error> { | ||||||
|     Parser { iter: s.chars(), src: s, current: (0, 0, 0) }.parse() |     Parser { iter: s.to_lowercase().chars(), src: &s.to_lowercase(), current: (0, 0, 0, 0) }.parse() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[cfg(test)] | ||||||
|  | mod tests { | ||||||
|  |     use super::*; | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn parse_seconds() { | ||||||
|  |         let interval = parse_duration("10 seconds").unwrap(); | ||||||
|  |  | ||||||
|  |         assert_eq!(interval.sec, 10); | ||||||
|  |         assert_eq!(interval.day, 0); | ||||||
|  |         assert_eq!(interval.month, 0); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn parse_minutes() { | ||||||
|  |         let interval = parse_duration("10 minutes").unwrap(); | ||||||
|  |  | ||||||
|  |         assert_eq!(interval.sec, 600); | ||||||
|  |         assert_eq!(interval.day, 0); | ||||||
|  |         assert_eq!(interval.month, 0); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn parse_hours() { | ||||||
|  |         let interval = parse_duration("10 hours").unwrap(); | ||||||
|  |  | ||||||
|  |         assert_eq!(interval.sec, 36_000); | ||||||
|  |         assert_eq!(interval.day, 0); | ||||||
|  |         assert_eq!(interval.month, 0); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn parse_days() { | ||||||
|  |         let interval = parse_duration("10 days").unwrap(); | ||||||
|  |  | ||||||
|  |         assert_eq!(interval.sec, 0); | ||||||
|  |         assert_eq!(interval.day, 10); | ||||||
|  |         assert_eq!(interval.month, 0); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn parse_weeks() { | ||||||
|  |         let interval = parse_duration("10 weeks").unwrap(); | ||||||
|  |  | ||||||
|  |         assert_eq!(interval.sec, 0); | ||||||
|  |         assert_eq!(interval.day, 70); | ||||||
|  |         assert_eq!(interval.month, 0); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn parse_months() { | ||||||
|  |         let interval = parse_duration("10 months").unwrap(); | ||||||
|  |  | ||||||
|  |         assert_eq!(interval.sec, 0); | ||||||
|  |         assert_eq!(interval.day, 0); | ||||||
|  |         assert_eq!(interval.month, 10); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn parse_years() { | ||||||
|  |         let interval = parse_duration("10 years").unwrap(); | ||||||
|  |  | ||||||
|  |         assert_eq!(interval.sec, 0); | ||||||
|  |         assert_eq!(interval.day, 0); | ||||||
|  |         assert_eq!(interval.month, 120); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn parse_case() { | ||||||
|  |         let interval = parse_duration("200 Seconds").unwrap(); | ||||||
|  |  | ||||||
|  |         assert_eq!(interval.sec, 200); | ||||||
|  |         assert_eq!(interval.day, 0); | ||||||
|  |         assert_eq!(interval.month, 0); | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										131
									
								
								src/main.rs
									
									
									
									
									
								
							
							
						
						| @@ -1,4 +1,5 @@ | |||||||
| #![feature(int_roundings)] | #![feature(int_roundings)] | ||||||
|  |  | ||||||
| #[macro_use] | #[macro_use] | ||||||
| extern crate lazy_static; | extern crate lazy_static; | ||||||
|  |  | ||||||
| @@ -17,20 +18,20 @@ use std::{ | |||||||
|     env, |     env, | ||||||
|     error::Error as StdError, |     error::Error as StdError, | ||||||
|     fmt::{Debug, Display, Formatter}, |     fmt::{Debug, Display, Formatter}, | ||||||
|     sync::atomic::AtomicBool, |     path::Path, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
| use dotenv::dotenv; | use log::{error, warn}; | ||||||
| use poise::serenity::model::{ | use poise::serenity_prelude::model::{ | ||||||
|     gateway::{Activity, GatewayIntents}, |     gateway::GatewayIntents, | ||||||
|     id::{GuildId, UserId}, |     id::{GuildId, UserId}, | ||||||
| }; | }; | ||||||
| use sqlx::{MySql, Pool}; | use sqlx::{MySql, Pool}; | ||||||
| use tokio::sync::{broadcast, broadcast::Sender, RwLock}; | use tokio::sync::{broadcast, broadcast::Sender, RwLock}; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
|     commands::{info_cmds, moderation_cmds, reminder_cmds, todo_cmds}, |     commands::{command_macro, info_cmds, moderation_cmds, reminder_cmds, todo_cmds}, | ||||||
|     consts::THEME_COLOR, |     consts::THEME_COLOR, | ||||||
|     event_handlers::listener, |     event_handlers::listener, | ||||||
|     hooks::all_checks, |     hooks::all_checks, | ||||||
| @@ -42,17 +43,17 @@ type Database = MySql; | |||||||
|  |  | ||||||
| type Error = Box<dyn std::error::Error + Send + Sync>; | type Error = Box<dyn std::error::Error + Send + Sync>; | ||||||
| type Context<'a> = poise::Context<'a, Data, Error>; | type Context<'a> = poise::Context<'a, Data, Error>; | ||||||
|  | type ApplicationContext<'a> = poise::ApplicationContext<'a, Data, Error>; | ||||||
|  |  | ||||||
| pub struct Data { | pub struct Data { | ||||||
|     database: Pool<Database>, |     database: Pool<Database>, | ||||||
|     http: reqwest::Client, |     http: reqwest::Client, | ||||||
|     recording_macros: RwLock<HashMap<(GuildId, UserId), CommandMacro<Data, Error>>>, |     recording_macros: RwLock<HashMap<(GuildId, UserId), CommandMacro<Data, Error>>>, | ||||||
|     popular_timezones: Vec<Tz>, |     popular_timezones: Vec<Tz>, | ||||||
|     is_loop_running: AtomicBool, |     _broadcast: Sender<()>, | ||||||
|     broadcast: Sender<()>, |  | ||||||
| } | } | ||||||
|  |  | ||||||
| impl std::fmt::Debug for Data { | impl Debug for Data { | ||||||
|     fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { |     fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { | ||||||
|         write!(f, "Data {{ .. }}") |         write!(f, "Data {{ .. }}") | ||||||
|     } |     } | ||||||
| @@ -74,7 +75,7 @@ impl Display for Ended { | |||||||
|  |  | ||||||
| impl StdError for Ended {} | impl StdError for Ended {} | ||||||
|  |  | ||||||
| #[tokio::main] | #[tokio::main(flavor = "multi_thread")] | ||||||
| async fn main() -> Result<(), Box<dyn StdError + Send + Sync>> { | async fn main() -> Result<(), Box<dyn StdError + Send + Sync>> { | ||||||
|     let (tx, mut rx) = broadcast::channel(16); |     let (tx, mut rx) = broadcast::channel(16); | ||||||
|  |  | ||||||
| @@ -87,7 +88,11 @@ async fn main() -> Result<(), Box<dyn StdError + Send + Sync>> { | |||||||
| async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | ||||||
|     env_logger::init(); |     env_logger::init(); | ||||||
|  |  | ||||||
|     dotenv()?; |     if Path::new("/etc/reminder-rs/config.env").exists() { | ||||||
|  |         dotenv::from_path("/etc/reminder-rs/config.env")?; | ||||||
|  |     } else { | ||||||
|  |         let _ = dotenv::dotenv(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     let discord_token = env::var("DISCORD_TOKEN").expect("Missing DISCORD_TOKEN from environment"); |     let discord_token = env::var("DISCORD_TOKEN").expect("Missing DISCORD_TOKEN from environment"); | ||||||
|  |  | ||||||
| @@ -102,13 +107,32 @@ async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | |||||||
|             moderation_cmds::timezone(), |             moderation_cmds::timezone(), | ||||||
|             poise::Command { |             poise::Command { | ||||||
|                 subcommands: vec![ |                 subcommands: vec![ | ||||||
|                     moderation_cmds::delete_macro(), |                     moderation_cmds::set_allowed_dm(), | ||||||
|                     moderation_cmds::finish_macro(), |                     moderation_cmds::unset_allowed_dm(), | ||||||
|                     moderation_cmds::list_macro(), |  | ||||||
|                     moderation_cmds::record_macro(), |  | ||||||
|                     moderation_cmds::run_macro(), |  | ||||||
|                 ], |                 ], | ||||||
|                 ..moderation_cmds::macro_base() |                 ..moderation_cmds::allowed_dm() | ||||||
|  |             }, | ||||||
|  |             poise::Command { | ||||||
|  |                 subcommands: vec![poise::Command { | ||||||
|  |                     subcommands: vec![ | ||||||
|  |                         moderation_cmds::set_ephemeral_confirmations(), | ||||||
|  |                         moderation_cmds::unset_ephemeral_confirmations(), | ||||||
|  |                     ], | ||||||
|  |                     ..moderation_cmds::ephemeral_confirmations() | ||||||
|  |                 }], | ||||||
|  |                 ..moderation_cmds::settings() | ||||||
|  |             }, | ||||||
|  |             moderation_cmds::webhook(), | ||||||
|  |             poise::Command { | ||||||
|  |                 subcommands: vec![ | ||||||
|  |                     command_macro::delete::delete_macro(), | ||||||
|  |                     command_macro::record::finish_macro(), | ||||||
|  |                     command_macro::list::list_macro(), | ||||||
|  |                     command_macro::record::record_macro(), | ||||||
|  |                     command_macro::run::run_macro(), | ||||||
|  |                     command_macro::migrate::migrate_macro(), | ||||||
|  |                 ], | ||||||
|  |                 ..command_macro::macro_base() | ||||||
|             }, |             }, | ||||||
|             reminder_cmds::pause(), |             reminder_cmds::pause(), | ||||||
|             reminder_cmds::offset(), |             reminder_cmds::offset(), | ||||||
| @@ -123,6 +147,7 @@ async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | |||||||
|                 ], |                 ], | ||||||
|                 ..reminder_cmds::timer_base() |                 ..reminder_cmds::timer_base() | ||||||
|             }, |             }, | ||||||
|  |             reminder_cmds::multiline(), | ||||||
|             reminder_cmds::remind(), |             reminder_cmds::remind(), | ||||||
|             poise::Command { |             poise::Command { | ||||||
|                 subcommands: vec![ |                 subcommands: vec![ | ||||||
| @@ -150,15 +175,36 @@ async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | |||||||
|         ], |         ], | ||||||
|         allowed_mentions: None, |         allowed_mentions: None, | ||||||
|         command_check: Some(|ctx| Box::pin(all_checks(ctx))), |         command_check: Some(|ctx| Box::pin(all_checks(ctx))), | ||||||
|         listener: |ctx, event, _framework, data| Box::pin(listener(ctx, event, data)), |         event_handler: |ctx, event, _framework, data| Box::pin(listener(ctx, event, data)), | ||||||
|  |         on_error: |error| { | ||||||
|  |             Box::pin(async move { | ||||||
|  |                 match error { | ||||||
|  |                     poise::FrameworkError::CommandCheckFailed { .. } => { | ||||||
|  |                         // suppress error | ||||||
|  |                     } | ||||||
|  |                     error => { | ||||||
|  |                         if let Err(e) = poise::builtins::on_error(error).await { | ||||||
|  |                             log::error!("Error while handling error: {}", e); | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             }) | ||||||
|  |         }, | ||||||
|         ..Default::default() |         ..Default::default() | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     let database = |     let database = | ||||||
|         Pool::connect(&env::var("DATABASE_URL").expect("No database URL provided")).await.unwrap(); |         Pool::connect(&env::var("DATABASE_URL").expect("No database URL provided")).await.unwrap(); | ||||||
|  |  | ||||||
|  |     sqlx::migrate!().run(&database).await?; | ||||||
|  |  | ||||||
|     let popular_timezones = sqlx::query!( |     let popular_timezones = sqlx::query!( | ||||||
|         "SELECT timezone FROM users GROUP BY timezone ORDER BY COUNT(timezone) DESC LIMIT 21" |         "SELECT IFNULL(timezone, 'UTC') AS timezone | ||||||
|  |         FROM users | ||||||
|  |         WHERE timezone IS NOT NULL | ||||||
|  |         GROUP BY timezone | ||||||
|  |         ORDER BY COUNT(timezone) DESC | ||||||
|  |         LIMIT 21" | ||||||
|     ) |     ) | ||||||
|     .fetch_all(&database) |     .fetch_all(&database) | ||||||
|     .await |     .await | ||||||
| @@ -167,29 +213,50 @@ async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | |||||||
|     .map(|t| t.timezone.parse::<Tz>().unwrap()) |     .map(|t| t.timezone.parse::<Tz>().unwrap()) | ||||||
|     .collect::<Vec<Tz>>(); |     .collect::<Vec<Tz>>(); | ||||||
|  |  | ||||||
|     poise::Framework::build() |     poise::Framework::builder() | ||||||
|         .token(discord_token) |         .token(discord_token) | ||||||
|         .user_data_setup(move |ctx, _bot, framework| { |         .setup(move |ctx, _bot, framework| { | ||||||
|             Box::pin(async move { |             Box::pin(async move { | ||||||
|                 ctx.set_activity(Activity::watching("for /remind")).await; |                 register_application_commands(ctx, framework, None).await.unwrap(); | ||||||
|  |  | ||||||
|                 register_application_commands( |                 let kill_tx = tx.clone(); | ||||||
|                     ctx, |                 let kill_recv = tx.subscribe(); | ||||||
|                     framework, |  | ||||||
|                     env::var("DEBUG_GUILD") |                 let ctx1 = ctx.clone(); | ||||||
|                         .map(|inner| GuildId(inner.parse().expect("DEBUG_GUILD not valid"))) |                 let ctx2 = ctx.clone(); | ||||||
|                         .ok(), |  | ||||||
|                 ) |                 let pool1 = database.clone(); | ||||||
|                 .await |                 let pool2 = database.clone(); | ||||||
|                 .unwrap(); |  | ||||||
|  |                 let run_settings = env::var("DONTRUN").unwrap_or_else(|_| "".to_string()); | ||||||
|  |  | ||||||
|  |                 if !run_settings.contains("postman") { | ||||||
|  |                     tokio::spawn(async move { | ||||||
|  |                         match postman::initialize(kill_recv, ctx1, &pool1).await { | ||||||
|  |                             Ok(_) => {} | ||||||
|  |                             Err(e) => { | ||||||
|  |                                 error!("postman exiting: {}", e); | ||||||
|  |                             } | ||||||
|  |                         }; | ||||||
|  |                     }); | ||||||
|  |                 } else { | ||||||
|  |                     warn!("Not running postman"); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 if !run_settings.contains("web") { | ||||||
|  |                     tokio::spawn(async move { | ||||||
|  |                         reminder_web::initialize(kill_tx, ctx2, pool2).await.unwrap(); | ||||||
|  |                     }); | ||||||
|  |                 } else { | ||||||
|  |                     warn!("Not running web"); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|                 Ok(Data { |                 Ok(Data { | ||||||
|                     http: reqwest::Client::new(), |                     http: reqwest::Client::new(), | ||||||
|                     database, |                     database, | ||||||
|                     popular_timezones, |                     popular_timezones, | ||||||
|                     recording_macros: Default::default(), |                     recording_macros: Default::default(), | ||||||
|                     is_loop_running: AtomicBool::new(false), |                     _broadcast: tx, | ||||||
|                     broadcast: tx, |  | ||||||
|                 }) |                 }) | ||||||
|             }) |             }) | ||||||
|         }) |         }) | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| use chrono::NaiveDateTime; | use chrono::NaiveDateTime; | ||||||
| use poise::serenity::model::channel::Channel; | use poise::serenity_prelude::model::channel::Channel; | ||||||
| use sqlx::MySqlPool; | use sqlx::MySqlPool; | ||||||
|  |  | ||||||
| pub struct ChannelData { | pub struct ChannelData { | ||||||
| @@ -22,9 +22,7 @@ impl ChannelData { | |||||||
|  |  | ||||||
|         if let Ok(c) = sqlx::query_as_unchecked!( |         if let Ok(c) = sqlx::query_as_unchecked!( | ||||||
|             Self, |             Self, | ||||||
|             " |             "SELECT id, name, nudge, blacklisted, webhook_id, webhook_token, paused, paused_until FROM channels WHERE channel = ?", | ||||||
| SELECT id, name, nudge, blacklisted, webhook_id, webhook_token, paused, paused_until FROM channels WHERE channel = ? |  | ||||||
|             ", |  | ||||||
|             channel_id |             channel_id | ||||||
|         ) |         ) | ||||||
|         .fetch_one(pool) |         .fetch_one(pool) | ||||||
| @@ -37,9 +35,7 @@ SELECT id, name, nudge, blacklisted, webhook_id, webhook_token, paused, paused_u | |||||||
|             let (guild_id, channel_name) = if let Some((a, b)) = props { (Some(a), Some(b)) } else { (None, None) }; |             let (guild_id, channel_name) = if let Some((a, b)) = props { (Some(a), Some(b)) } else { (None, None) }; | ||||||
|  |  | ||||||
|             sqlx::query!( |             sqlx::query!( | ||||||
|                 " |                 "INSERT IGNORE INTO channels (channel, name, guild_id) VALUES (?, ?, (SELECT id FROM guilds WHERE guild = ?))", | ||||||
| INSERT IGNORE INTO channels (channel, name, guild_id) VALUES (?, ?, (SELECT id FROM guilds WHERE guild = ?)) |  | ||||||
|                 ", |  | ||||||
|                 channel_id, |                 channel_id, | ||||||
|                 channel_name, |                 channel_name, | ||||||
|                 guild_id |                 guild_id | ||||||
|   | |||||||
| @@ -1,15 +1,16 @@ | |||||||
| use poise::serenity::model::{ | use poise::serenity_prelude::model::{ | ||||||
|     id::GuildId, interactions::application_command::ApplicationCommandInteractionDataOption, |     application::interaction::application_command::CommandDataOption, id::GuildId, | ||||||
| }; | }; | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
|  | use serde_json::Value; | ||||||
|  |  | ||||||
| use crate::{Context, Data, Error}; | use crate::{Context, Data, Error}; | ||||||
|  |  | ||||||
| fn default_none<U, E>() -> Option< | type Func<U, E> = for<'a> fn( | ||||||
|     for<'a> fn( |     poise::ApplicationContext<'a, U, E>, | ||||||
|         poise::ApplicationContext<'a, U, E>, | ) -> poise::BoxFuture<'a, Result<(), poise::FrameworkError<'a, U, E>>>; | ||||||
|     ) -> poise::BoxFuture<'a, Result<(), poise::FrameworkError<'a, U, E>>>, |  | ||||||
| > { | fn default_none<U, E>() -> Option<Func<U, E>> { | ||||||
|     None |     None | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -17,13 +18,9 @@ fn default_none<U, E>() -> Option< | |||||||
| pub struct RecordedCommand<U, E> { | pub struct RecordedCommand<U, E> { | ||||||
|     #[serde(skip)] |     #[serde(skip)] | ||||||
|     #[serde(default = "default_none::<U, E>")] |     #[serde(default = "default_none::<U, E>")] | ||||||
|     pub action: Option< |     pub action: Option<Func<U, E>>, | ||||||
|         for<'a> fn( |  | ||||||
|             poise::ApplicationContext<'a, U, E>, |  | ||||||
|         ) -> poise::BoxFuture<'a, Result<(), poise::FrameworkError<'a, U, E>>>, |  | ||||||
|     >, |  | ||||||
|     pub command_name: String, |     pub command_name: String, | ||||||
|     pub options: Vec<ApplicationCommandInteractionDataOption>, |     pub options: Vec<CommandDataOption>, | ||||||
| } | } | ||||||
|  |  | ||||||
| pub struct CommandMacro<U, E> { | pub struct CommandMacro<U, E> { | ||||||
| @@ -33,6 +30,13 @@ pub struct CommandMacro<U, E> { | |||||||
|     pub commands: Vec<RecordedCommand<U, E>>, |     pub commands: Vec<RecordedCommand<U, E>>, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | pub struct RawCommandMacro { | ||||||
|  |     pub guild_id: GuildId, | ||||||
|  |     pub name: String, | ||||||
|  |     pub description: Option<String>, | ||||||
|  |     pub commands: Value, | ||||||
|  | } | ||||||
|  |  | ||||||
| pub async fn guild_command_macro( | pub async fn guild_command_macro( | ||||||
|     ctx: &Context<'_>, |     ctx: &Context<'_>, | ||||||
|     name: &str, |     name: &str, | ||||||
| @@ -59,7 +63,7 @@ SELECT * FROM macro WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?) AND | |||||||
|             .iter() |             .iter() | ||||||
|             .find(|c| c.identifying_name == recorded_command.command_name); |             .find(|c| c.identifying_name == recorded_command.command_name); | ||||||
|  |  | ||||||
|         recorded_command.action = command.map(|c| c.slash_action).flatten().clone(); |         recorded_command.action = command.map(|c| c.slash_action).flatten(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     let command_macro = CommandMacro { |     let command_macro = CommandMacro { | ||||||
|   | |||||||
							
								
								
									
										48
									
								
								src/models/guild_data.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,48 @@ | |||||||
|  | use poise::serenity_prelude::GuildId; | ||||||
|  | use sqlx::MySqlPool; | ||||||
|  |  | ||||||
|  | pub struct GuildData { | ||||||
|  |     pub ephemeral_confirmations: bool, | ||||||
|  |     pub id: u32, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl GuildData { | ||||||
|  |     pub async fn from_guild( | ||||||
|  |         guild_id: GuildId, | ||||||
|  |         pool: &MySqlPool, | ||||||
|  |     ) -> Result<Self, Box<dyn std::error::Error + Sync + Send>> { | ||||||
|  |         if let Ok(c) = sqlx::query_as_unchecked!( | ||||||
|  |             Self, | ||||||
|  |             "SELECT id, ephemeral_confirmations FROM guilds WHERE guild = ?", | ||||||
|  |             guild_id.0 | ||||||
|  |         ) | ||||||
|  |         .fetch_one(pool) | ||||||
|  |         .await | ||||||
|  |         { | ||||||
|  |             Ok(c) | ||||||
|  |         } else { | ||||||
|  |             sqlx::query!("INSERT IGNORE INTO guilds (guild) VALUES (?)", guild_id.0) | ||||||
|  |                 .execute(&pool.clone()) | ||||||
|  |                 .await?; | ||||||
|  |  | ||||||
|  |             Ok(sqlx::query_as_unchecked!( | ||||||
|  |                 Self, | ||||||
|  |                 "SELECT id, ephemeral_confirmations FROM guilds WHERE guild = ?", | ||||||
|  |                 guild_id.0 | ||||||
|  |             ) | ||||||
|  |             .fetch_one(pool) | ||||||
|  |             .await?) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub async fn commit_changes(&self, pool: &MySqlPool) { | ||||||
|  |         sqlx::query!( | ||||||
|  |             "UPDATE guilds SET ephemeral_confirmations = ? WHERE id = ?", | ||||||
|  |             self.ephemeral_confirmations, | ||||||
|  |             self.id | ||||||
|  |         ) | ||||||
|  |         .execute(pool) | ||||||
|  |         .await | ||||||
|  |         .unwrap(); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,14 +1,15 @@ | |||||||
| pub mod channel_data; | pub mod channel_data; | ||||||
| pub mod command_macro; | pub mod command_macro; | ||||||
|  | pub mod guild_data; | ||||||
| pub mod reminder; | pub mod reminder; | ||||||
| pub mod timer; | pub mod timer; | ||||||
| pub mod user_data; | pub mod user_data; | ||||||
|  |  | ||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
| use poise::serenity::{async_trait, model::id::UserId}; | use poise::serenity_prelude::{async_trait, model::id::UserId, ChannelType}; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
|     models::{channel_data::ChannelData, user_data::UserData}, |     models::{channel_data::ChannelData, guild_data::GuildData, user_data::UserData}, | ||||||
|     CommandMacro, Context, Data, Error, GuildId, |     CommandMacro, Context, Data, Error, GuildId, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @@ -18,6 +19,8 @@ pub trait CtxData { | |||||||
|  |  | ||||||
|     async fn author_data(&self) -> Result<UserData, Error>; |     async fn author_data(&self) -> Result<UserData, Error>; | ||||||
|  |  | ||||||
|  |     async fn guild_data(&self) -> Option<Result<GuildData, Error>>; | ||||||
|  |  | ||||||
|     async fn timezone(&self) -> Tz; |     async fn timezone(&self) -> Tz; | ||||||
|  |  | ||||||
|     async fn channel_data(&self) -> Result<ChannelData, Error>; |     async fn channel_data(&self) -> Result<ChannelData, Error>; | ||||||
| @@ -27,15 +30,21 @@ pub trait CtxData { | |||||||
|  |  | ||||||
| #[async_trait] | #[async_trait] | ||||||
| impl CtxData for Context<'_> { | impl CtxData for Context<'_> { | ||||||
|     async fn user_data<U: Into<UserId> + Send>( |     async fn user_data<U: Into<UserId> + Send>(&self, user_id: U) -> Result<UserData, Error> { | ||||||
|         &self, |         UserData::from_user(user_id, &self.serenity_context(), &self.data().database).await | ||||||
|         user_id: U, |  | ||||||
|     ) -> Result<UserData, Box<dyn std::error::Error + Sync + Send>> { |  | ||||||
|         UserData::from_user(user_id, &self.discord(), &self.data().database).await |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     async fn author_data(&self) -> Result<UserData, Box<dyn std::error::Error + Sync + Send>> { |     async fn author_data(&self) -> Result<UserData, Error> { | ||||||
|         UserData::from_user(&self.author().id, &self.discord(), &self.data().database).await |         UserData::from_user(&self.author().id, &self.serenity_context(), &self.data().database) | ||||||
|  |             .await | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async fn guild_data(&self) -> Option<Result<GuildData, Error>> { | ||||||
|  |         if let Some(guild_id) = self.guild_id() { | ||||||
|  |             Some(GuildData::from_guild(guild_id, &self.data().database).await) | ||||||
|  |         } else { | ||||||
|  |             None | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     async fn timezone(&self) -> Tz { |     async fn timezone(&self) -> Tz { | ||||||
| @@ -43,7 +52,20 @@ impl CtxData for Context<'_> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     async fn channel_data(&self) -> Result<ChannelData, Box<dyn std::error::Error + Sync + Send>> { |     async fn channel_data(&self) -> Result<ChannelData, Box<dyn std::error::Error + Sync + Send>> { | ||||||
|         let channel = self.channel_id().to_channel_cached(&self.discord()).unwrap(); |         // If we're in a thread, get the parent channel. | ||||||
|  |         let recv_channel = self.channel_id().to_channel(&self).await?; | ||||||
|  |  | ||||||
|  |         let channel = match recv_channel.guild() { | ||||||
|  |             Some(guild_channel) => { | ||||||
|  |                 if guild_channel.kind == ChannelType::PublicThread { | ||||||
|  |                     guild_channel.parent_id.unwrap().to_channel_cached(&self).unwrap() | ||||||
|  |                 } else { | ||||||
|  |                     self.channel_id().to_channel_cached(&self).unwrap() | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             None => self.channel_id().to_channel_cached(&self).unwrap(), | ||||||
|  |         }; | ||||||
|  |  | ||||||
|         ChannelData::from_channel(&channel, &self.data().database).await |         ChannelData::from_channel(&channel, &self.data().database).await | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -2,14 +2,14 @@ use std::{collections::HashSet, fmt::Display}; | |||||||
|  |  | ||||||
| use chrono::{Duration, NaiveDateTime, Utc}; | use chrono::{Duration, NaiveDateTime, Utc}; | ||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
| use poise::serenity::{ | use poise::serenity_prelude::{ | ||||||
|     http::CacheHttp, |     http::CacheHttp, | ||||||
|     model::{ |     model::{ | ||||||
|         channel::GuildChannel, |         channel::GuildChannel, | ||||||
|         id::{ChannelId, GuildId, UserId}, |         id::{ChannelId, GuildId, UserId}, | ||||||
|         webhook::Webhook, |         webhook::Webhook, | ||||||
|     }, |     }, | ||||||
|     Result as SerenityResult, |     ChannelType, Result as SerenityResult, | ||||||
| }; | }; | ||||||
| use sqlx::MySqlPool; | use sqlx::MySqlPool; | ||||||
|  |  | ||||||
| @@ -51,9 +51,11 @@ pub struct ReminderBuilder { | |||||||
|     pool: MySqlPool, |     pool: MySqlPool, | ||||||
|     uid: String, |     uid: String, | ||||||
|     channel: u32, |     channel: u32, | ||||||
|  |     thread_id: Option<u64>, | ||||||
|     utc_time: NaiveDateTime, |     utc_time: NaiveDateTime, | ||||||
|     timezone: String, |     timezone: String, | ||||||
|     interval_secs: Option<i64>, |     interval_seconds: Option<i64>, | ||||||
|  |     interval_days: Option<i64>, | ||||||
|     interval_months: Option<i64>, |     interval_months: Option<i64>, | ||||||
|     expires: Option<NaiveDateTime>, |     expires: Option<NaiveDateTime>, | ||||||
|     content: String, |     content: String, | ||||||
| @@ -87,6 +89,7 @@ INSERT INTO reminders ( | |||||||
|     `utc_time`, |     `utc_time`, | ||||||
|     `timezone`, |     `timezone`, | ||||||
|     `interval_seconds`, |     `interval_seconds`, | ||||||
|  |     `interval_days`, | ||||||
|     `interval_months`, |     `interval_months`, | ||||||
|     `expires`, |     `expires`, | ||||||
|     `content`, |     `content`, | ||||||
| @@ -106,6 +109,7 @@ INSERT INTO reminders ( | |||||||
|     ?, |     ?, | ||||||
|     ?, |     ?, | ||||||
|     ?, |     ?, | ||||||
|  |     ?, | ||||||
|     ? |     ? | ||||||
| ) | ) | ||||||
|             ", |             ", | ||||||
| @@ -113,7 +117,8 @@ INSERT INTO reminders ( | |||||||
|                         self.channel, |                         self.channel, | ||||||
|                         utc_time, |                         utc_time, | ||||||
|                         self.timezone, |                         self.timezone, | ||||||
|                         self.interval_secs, |                         self.interval_seconds, | ||||||
|  |                         self.interval_days, | ||||||
|                         self.interval_months, |                         self.interval_months, | ||||||
|                         self.expires, |                         self.expires, | ||||||
|                         self.content, |                         self.content, | ||||||
| @@ -126,7 +131,7 @@ INSERT INTO reminders ( | |||||||
|                     .await |                     .await | ||||||
|                     .unwrap(); |                     .unwrap(); | ||||||
|  |  | ||||||
|                     Ok(Reminder::from_uid(&self.pool, self.uid).await.unwrap()) |                     Ok(Reminder::from_uid(&self.pool, &self.uid).await.unwrap()) | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
| @@ -175,17 +180,15 @@ impl<'a> MultiReminderBuilder<'a> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn time<T: Into<i64>>(mut self, time: T) -> Self { |     pub fn time<T: Into<i64>>(mut self, time: T) -> Self { | ||||||
|         self.utc_time = NaiveDateTime::from_timestamp(time.into(), 0); |         if let Some(utc_time) = NaiveDateTime::from_timestamp_opt(time.into(), 0) { | ||||||
|  |             self.utc_time = utc_time; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         self |         self | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn expires<T: Into<i64>>(mut self, time: Option<T>) -> Self { |     pub fn expires<T: Into<i64>>(mut self, time: Option<T>) -> Self { | ||||||
|         if let Some(t) = time { |         self.expires = time.map(|t| NaiveDateTime::from_timestamp_opt(t.into(), 0)).flatten(); | ||||||
|             self.expires = Some(NaiveDateTime::from_timestamp(t.into(), 0)); |  | ||||||
|         } else { |  | ||||||
|             self.expires = None; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         self |         self | ||||||
|     } |     } | ||||||
| @@ -207,32 +210,42 @@ impl<'a> MultiReminderBuilder<'a> { | |||||||
|         self.scopes = scopes; |         self.scopes = scopes; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub async fn build(self) -> (HashSet<ReminderError>, HashSet<ReminderScope>) { |     pub async fn build(self) -> (HashSet<ReminderError>, HashSet<(Reminder, ReminderScope)>) { | ||||||
|         let mut errors = HashSet::new(); |         let mut errors = HashSet::new(); | ||||||
|  |  | ||||||
|         let mut ok_locs = HashSet::new(); |         let mut ok_locs = HashSet::new(); | ||||||
|  |  | ||||||
|         if self.interval.map_or(false, |i| ((i.sec + i.month * 30 * DAY) as i64) < *MIN_INTERVAL) { |         if self | ||||||
|  |             .interval | ||||||
|  |             .map_or(false, |i| ((i.sec + i.day * DAY + i.month * 30 * DAY) as i64) < *MIN_INTERVAL) | ||||||
|  |         { | ||||||
|             errors.insert(ReminderError::ShortInterval); |             errors.insert(ReminderError::ShortInterval); | ||||||
|         } else if self.interval.map_or(false, |i| ((i.sec + i.month * 30 * DAY) as i64) > *MAX_TIME) |         } else if self | ||||||
|  |             .interval | ||||||
|  |             .map_or(false, |i| ((i.sec + i.day * DAY + i.month * 30 * DAY) as i64) > *MAX_TIME) | ||||||
|         { |         { | ||||||
|             errors.insert(ReminderError::LongInterval); |             errors.insert(ReminderError::LongInterval); | ||||||
|         } else { |         } else { | ||||||
|             for scope in self.scopes { |             for scope in self.scopes { | ||||||
|  |                 let thread_id = None; | ||||||
|                 let db_channel_id = match scope { |                 let db_channel_id = match scope { | ||||||
|                     ReminderScope::User(user_id) => { |                     ReminderScope::User(user_id) => { | ||||||
|                         if let Ok(user) = UserId(user_id).to_user(&self.ctx.discord()).await { |                         if let Ok(user) = UserId(user_id).to_user(&self.ctx).await { | ||||||
|                             let user_data = UserData::from_user( |                             let user_data = UserData::from_user( | ||||||
|                                 &user, |                                 &user, | ||||||
|                                 &self.ctx.discord(), |                                 &self.ctx.serenity_context(), | ||||||
|                                 &self.ctx.data().database, |                                 &self.ctx.data().database, | ||||||
|                             ) |                             ) | ||||||
|                             .await |                             .await | ||||||
|                             .unwrap(); |                             .unwrap(); | ||||||
|  |  | ||||||
|                             if let Some(guild_id) = self.guild_id { |                             if let Some(guild_id) = self.guild_id { | ||||||
|                                 if guild_id.member(&self.ctx.discord(), user).await.is_err() { |                                 if guild_id.member(&self.ctx, user).await.is_err() { | ||||||
|                                     Err(ReminderError::InvalidTag) |                                     Err(ReminderError::InvalidTag) | ||||||
|  |                                 } else if self.set_by.map_or(true, |i| i != user_data.id) | ||||||
|  |                                     && !user_data.allowed_dm | ||||||
|  |                                 { | ||||||
|  |                                     Err(ReminderError::UserBlockedDm) | ||||||
|                                 } else { |                                 } else { | ||||||
|                                     Ok(user_data.dm_channel) |                                     Ok(user_data.dm_channel) | ||||||
|                                 } |                                 } | ||||||
| @@ -244,27 +257,36 @@ impl<'a> MultiReminderBuilder<'a> { | |||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                     ReminderScope::Channel(channel_id) => { |                     ReminderScope::Channel(channel_id) => { | ||||||
|                         let channel = |                         let channel = ChannelId(channel_id).to_channel(&self.ctx).await.unwrap(); | ||||||
|                             ChannelId(channel_id).to_channel(&self.ctx.discord()).await.unwrap(); |  | ||||||
|  |  | ||||||
|                         if let Some(guild_channel) = channel.clone().guild() { |                         if let Some(mut guild_channel) = channel.clone().guild() { | ||||||
|                             if Some(guild_channel.guild_id) != self.guild_id { |                             if Some(guild_channel.guild_id) != self.guild_id { | ||||||
|                                 Err(ReminderError::InvalidTag) |                                 Err(ReminderError::InvalidTag) | ||||||
|                             } else { |                             } else { | ||||||
|                                 let mut channel_data = |                                 let mut channel_data = if guild_channel.kind | ||||||
|                                     ChannelData::from_channel(&channel, &self.ctx.data().database) |                                     == ChannelType::PublicThread | ||||||
|  |                                 { | ||||||
|  |                                     // fixme jesus christ | ||||||
|  |                                     let parent = guild_channel | ||||||
|  |                                         .parent_id | ||||||
|  |                                         .unwrap() | ||||||
|  |                                         .to_channel(&self.ctx) | ||||||
|                                         .await |                                         .await | ||||||
|                                         .unwrap(); |                                         .unwrap(); | ||||||
|  |                                     guild_channel = parent.clone().guild().unwrap(); | ||||||
|  |                                     ChannelData::from_channel(&parent, &self.ctx.data().database) | ||||||
|  |                                         .await | ||||||
|  |                                         .unwrap() | ||||||
|  |                                 } else { | ||||||
|  |                                     ChannelData::from_channel(&channel, &self.ctx.data().database) | ||||||
|  |                                         .await | ||||||
|  |                                         .unwrap() | ||||||
|  |                                 }; | ||||||
|  |  | ||||||
|                                 if channel_data.webhook_id.is_none() |                                 if channel_data.webhook_id.is_none() | ||||||
|                                     || channel_data.webhook_token.is_none() |                                     || channel_data.webhook_token.is_none() | ||||||
|                                 { |                                 { | ||||||
|                                     match create_webhook( |                                     match create_webhook(&self.ctx, guild_channel, "Reminder").await | ||||||
|                                         &self.ctx.discord(), |  | ||||||
|                                         guild_channel, |  | ||||||
|                                         "Reminder", |  | ||||||
|                                     ) |  | ||||||
|                                     .await |  | ||||||
|                                     { |                                     { | ||||||
|                                         Ok(webhook) => { |                                         Ok(webhook) => { | ||||||
|                                             channel_data.webhook_id = |                                             channel_data.webhook_id = | ||||||
| @@ -296,9 +318,11 @@ impl<'a> MultiReminderBuilder<'a> { | |||||||
|                             pool: self.ctx.data().database.clone(), |                             pool: self.ctx.data().database.clone(), | ||||||
|                             uid: generate_uid(), |                             uid: generate_uid(), | ||||||
|                             channel: c, |                             channel: c, | ||||||
|  |                             thread_id, | ||||||
|                             utc_time: self.utc_time, |                             utc_time: self.utc_time, | ||||||
|                             timezone: self.timezone.to_string(), |                             timezone: self.timezone.to_string(), | ||||||
|                             interval_secs: self.interval.map(|i| i.sec as i64), |                             interval_seconds: self.interval.map(|i| i.sec as i64), | ||||||
|  |                             interval_days: self.interval.map(|i| i.day as i64), | ||||||
|                             interval_months: self.interval.map(|i| i.month as i64), |                             interval_months: self.interval.map(|i| i.month as i64), | ||||||
|                             expires: self.expires, |                             expires: self.expires, | ||||||
|                             content: self.content.content.clone(), |                             content: self.content.content.clone(), | ||||||
| @@ -309,8 +333,8 @@ impl<'a> MultiReminderBuilder<'a> { | |||||||
|                         }; |                         }; | ||||||
|  |  | ||||||
|                         match builder.build().await { |                         match builder.build().await { | ||||||
|                             Ok(_) => { |                             Ok(r) => { | ||||||
|                                 ok_locs.insert(scope); |                                 ok_locs.insert((r, scope)); | ||||||
|                             } |                             } | ||||||
|                             Err(e) => { |                             Err(e) => { | ||||||
|                                 errors.insert(e); |                                 errors.insert(e); | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ pub enum ReminderError { | |||||||
|     PastTime, |     PastTime, | ||||||
|     ShortInterval, |     ShortInterval, | ||||||
|     InvalidTag, |     InvalidTag, | ||||||
|  |     UserBlockedDm, | ||||||
|     DiscordError(String), |     DiscordError(String), | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -30,6 +31,9 @@ impl ToString for ReminderError { | |||||||
|             ReminderError::InvalidTag => { |             ReminderError::InvalidTag => { | ||||||
|                 "Couldn't find a location by your tag. Your tag must be either a channel or a user (not a role)".to_string() |                 "Couldn't find a location by your tag. Your tag must be either a channel or a user (not a role)".to_string() | ||||||
|             } |             } | ||||||
|  |             ReminderError::UserBlockedDm => { | ||||||
|  |                 "User has DM reminders disabled".to_string() | ||||||
|  |             } | ||||||
|             ReminderError::DiscordError(s) => format!("A Discord error occurred: **{}**", s), |             ReminderError::DiscordError(s) => format!("A Discord error occurred: **{}**", s), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| use poise::serenity::model::id::ChannelId; | use poise::serenity_prelude::model::id::ChannelId; | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
| use serde_repr::*; | use serde_repr::*; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -4,11 +4,13 @@ pub mod errors; | |||||||
| mod helper; | mod helper; | ||||||
| pub mod look_flags; | pub mod look_flags; | ||||||
|  |  | ||||||
| use chrono::{NaiveDateTime, TimeZone}; | use std::hash::{Hash, Hasher}; | ||||||
|  |  | ||||||
|  | use chrono::{DateTime, NaiveDateTime, Utc}; | ||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
| use poise::{ | use poise::serenity_prelude::{ | ||||||
|     serenity::model::id::{ChannelId, GuildId, UserId}, |     model::id::{ChannelId, GuildId, UserId}, | ||||||
|     serenity_prelude::Cache, |     Cache, | ||||||
| }; | }; | ||||||
| use sqlx::Executor; | use sqlx::Executor; | ||||||
|  |  | ||||||
| @@ -22,8 +24,9 @@ pub struct Reminder { | |||||||
|     pub id: u32, |     pub id: u32, | ||||||
|     pub uid: String, |     pub uid: String, | ||||||
|     pub channel: u64, |     pub channel: u64, | ||||||
|     pub utc_time: NaiveDateTime, |     pub utc_time: DateTime<Utc>, | ||||||
|     pub interval_seconds: Option<u32>, |     pub interval_seconds: Option<u32>, | ||||||
|  |     pub interval_days: Option<u32>, | ||||||
|     pub interval_months: Option<u32>, |     pub interval_months: Option<u32>, | ||||||
|     pub expires: Option<NaiveDateTime>, |     pub expires: Option<NaiveDateTime>, | ||||||
|     pub enabled: bool, |     pub enabled: bool, | ||||||
| @@ -32,11 +35,22 @@ pub struct Reminder { | |||||||
|     pub set_by: Option<u64>, |     pub set_by: Option<u64>, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | impl Hash for Reminder { | ||||||
|  |     fn hash<H: Hasher>(&self, state: &mut H) { | ||||||
|  |         self.uid.hash(state); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl PartialEq<Self> for Reminder { | ||||||
|  |     fn eq(&self, other: &Self) -> bool { | ||||||
|  |         self.uid == other.uid | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl Eq for Reminder {} | ||||||
|  |  | ||||||
| impl Reminder { | impl Reminder { | ||||||
|     pub async fn from_uid( |     pub async fn from_uid(pool: impl Executor<'_, Database = Database>, uid: &str) -> Option<Self> { | ||||||
|         pool: impl Executor<'_, Database = Database>, |  | ||||||
|         uid: String, |  | ||||||
|     ) -> Option<Self> { |  | ||||||
|         sqlx::query_as_unchecked!( |         sqlx::query_as_unchecked!( | ||||||
|             Self, |             Self, | ||||||
|             " |             " | ||||||
| @@ -46,6 +60,7 @@ SELECT | |||||||
|     channels.channel, |     channels.channel, | ||||||
|     reminders.utc_time, |     reminders.utc_time, | ||||||
|     reminders.interval_seconds, |     reminders.interval_seconds, | ||||||
|  |     reminders.interval_days, | ||||||
|     reminders.interval_months, |     reminders.interval_months, | ||||||
|     reminders.expires, |     reminders.expires, | ||||||
|     reminders.enabled, |     reminders.enabled, | ||||||
| @@ -72,14 +87,7 @@ WHERE | |||||||
|         .ok() |         .ok() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub async fn from_channel<C: Into<ChannelId>>( |     pub async fn from_id(pool: impl Executor<'_, Database = Database>, id: u32) -> Option<Self> { | ||||||
|         pool: impl Executor<'_, Database = Database>, |  | ||||||
|         channel_id: C, |  | ||||||
|         flags: &LookFlags, |  | ||||||
|     ) -> Vec<Self> { |  | ||||||
|         let enabled = if flags.show_disabled { "0,1" } else { "1" }; |  | ||||||
|         let channel_id = channel_id.into(); |  | ||||||
|  |  | ||||||
|         sqlx::query_as_unchecked!( |         sqlx::query_as_unchecked!( | ||||||
|             Self, |             Self, | ||||||
|             " |             " | ||||||
| @@ -89,6 +97,7 @@ SELECT | |||||||
|     channels.channel, |     channels.channel, | ||||||
|     reminders.utc_time, |     reminders.utc_time, | ||||||
|     reminders.interval_seconds, |     reminders.interval_seconds, | ||||||
|  |     reminders.interval_days, | ||||||
|     reminders.interval_months, |     reminders.interval_months, | ||||||
|     reminders.expires, |     reminders.expires, | ||||||
|     reminders.enabled, |     reminders.enabled, | ||||||
| @@ -106,6 +115,51 @@ LEFT JOIN | |||||||
| ON | ON | ||||||
|     reminders.set_by = users.id |     reminders.set_by = users.id | ||||||
| WHERE | WHERE | ||||||
|  |     reminders.id = ? | ||||||
|  |             ", | ||||||
|  |             id | ||||||
|  |         ) | ||||||
|  |         .fetch_one(pool) | ||||||
|  |         .await | ||||||
|  |         .ok() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub async fn from_channel<C: Into<ChannelId>>( | ||||||
|  |         pool: impl Executor<'_, Database = Database>, | ||||||
|  |         channel_id: C, | ||||||
|  |         flags: &LookFlags, | ||||||
|  |     ) -> Vec<Self> { | ||||||
|  |         let enabled = if flags.show_disabled { "0,1" } else { "1" }; | ||||||
|  |         let channel_id = channel_id.into(); | ||||||
|  |  | ||||||
|  |         sqlx::query_as_unchecked!( | ||||||
|  |             Self, | ||||||
|  |             " | ||||||
|  | SELECT | ||||||
|  |     reminders.id, | ||||||
|  |     reminders.uid, | ||||||
|  |     channels.channel, | ||||||
|  |     reminders.utc_time, | ||||||
|  |     reminders.interval_seconds, | ||||||
|  |     reminders.interval_days, | ||||||
|  |     reminders.interval_months, | ||||||
|  |     reminders.expires, | ||||||
|  |     reminders.enabled, | ||||||
|  |     reminders.content, | ||||||
|  |     reminders.embed_description, | ||||||
|  |     users.user AS set_by | ||||||
|  | FROM | ||||||
|  |     reminders | ||||||
|  | INNER JOIN | ||||||
|  |     channels | ||||||
|  | ON | ||||||
|  |     reminders.channel_id = channels.id | ||||||
|  | LEFT JOIN | ||||||
|  |     users | ||||||
|  | ON | ||||||
|  |     reminders.set_by = users.id | ||||||
|  | WHERE | ||||||
|  |     `status` = 'pending' AND | ||||||
|     channels.channel = ? AND |     channels.channel = ? AND | ||||||
|     FIND_IN_SET(reminders.enabled, ?) |     FIND_IN_SET(reminders.enabled, ?) | ||||||
| ORDER BY | ORDER BY | ||||||
| @@ -146,6 +200,7 @@ SELECT | |||||||
|     channels.channel, |     channels.channel, | ||||||
|     reminders.utc_time, |     reminders.utc_time, | ||||||
|     reminders.interval_seconds, |     reminders.interval_seconds, | ||||||
|  |     reminders.interval_days, | ||||||
|     reminders.interval_months, |     reminders.interval_months, | ||||||
|     reminders.expires, |     reminders.expires, | ||||||
|     reminders.enabled, |     reminders.enabled, | ||||||
| @@ -163,6 +218,7 @@ LEFT JOIN | |||||||
| ON | ON | ||||||
|     reminders.set_by = users.id |     reminders.set_by = users.id | ||||||
| WHERE | WHERE | ||||||
|  |     `status` = 'pending' AND | ||||||
|     FIND_IN_SET(channels.channel, ?) |     FIND_IN_SET(channels.channel, ?) | ||||||
|                 ", |                 ", | ||||||
|                     channels |                     channels | ||||||
| @@ -179,6 +235,7 @@ SELECT | |||||||
|     channels.channel, |     channels.channel, | ||||||
|     reminders.utc_time, |     reminders.utc_time, | ||||||
|     reminders.interval_seconds, |     reminders.interval_seconds, | ||||||
|  |     reminders.interval_days, | ||||||
|     reminders.interval_months, |     reminders.interval_months, | ||||||
|     reminders.expires, |     reminders.expires, | ||||||
|     reminders.enabled, |     reminders.enabled, | ||||||
| @@ -196,6 +253,7 @@ LEFT JOIN | |||||||
| ON | ON | ||||||
|     reminders.set_by = users.id |     reminders.set_by = users.id | ||||||
| WHERE | WHERE | ||||||
|  |     `status` = 'pending' AND | ||||||
|     channels.guild_id = (SELECT id FROM guilds WHERE guild = ?) |     channels.guild_id = (SELECT id FROM guilds WHERE guild = ?) | ||||||
|                 ", |                 ", | ||||||
|                     guild_id.as_u64() |                     guild_id.as_u64() | ||||||
| @@ -213,6 +271,7 @@ SELECT | |||||||
|     channels.channel, |     channels.channel, | ||||||
|     reminders.utc_time, |     reminders.utc_time, | ||||||
|     reminders.interval_seconds, |     reminders.interval_seconds, | ||||||
|  |     reminders.interval_days, | ||||||
|     reminders.interval_months, |     reminders.interval_months, | ||||||
|     reminders.expires, |     reminders.expires, | ||||||
|     reminders.enabled, |     reminders.enabled, | ||||||
| @@ -230,6 +289,7 @@ LEFT JOIN | |||||||
| ON | ON | ||||||
|     reminders.set_by = users.id |     reminders.set_by = users.id | ||||||
| WHERE | WHERE | ||||||
|  |     `status` = 'pending' AND | ||||||
|     channels.id = (SELECT dm_channel FROM users WHERE user = ?) |     channels.id = (SELECT dm_channel FROM users WHERE user = ?) | ||||||
|             ", |             ", | ||||||
|                 user.as_u64() |                 user.as_u64() | ||||||
| @@ -240,6 +300,16 @@ WHERE | |||||||
|         .unwrap() |         .unwrap() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     pub async fn delete( | ||||||
|  |         &self, | ||||||
|  |         db: impl Executor<'_, Database = Database>, | ||||||
|  |     ) -> Result<(), sqlx::Error> { | ||||||
|  |         sqlx::query!("UPDATE reminders SET `status` = 'deleted' WHERE uid = ?", self.uid) | ||||||
|  |             .execute(db) | ||||||
|  |             .await | ||||||
|  |             .map(|_| ()) | ||||||
|  |     } | ||||||
|  |  | ||||||
|     pub fn display_content(&self) -> &str { |     pub fn display_content(&self) -> &str { | ||||||
|         if self.content.is_empty() { |         if self.content.is_empty() { | ||||||
|             &self.embed_description |             &self.embed_description | ||||||
| @@ -254,33 +324,32 @@ WHERE | |||||||
|             count + 1, |             count + 1, | ||||||
|             self.display_content(), |             self.display_content(), | ||||||
|             self.channel, |             self.channel, | ||||||
|             timezone |             self.utc_time.with_timezone(timezone).format("%Y-%m-%d %H:%M:%S") | ||||||
|                 .timestamp(self.utc_time.timestamp(), 0) |  | ||||||
|                 .format("%Y-%m-%d %H:%M:%S") |  | ||||||
|                 .to_string() |  | ||||||
|         ) |         ) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn display(&self, flags: &LookFlags, timezone: &Tz) -> String { |     pub fn display(&self, flags: &LookFlags, timezone: &Tz) -> String { | ||||||
|         let time_display = match flags.time_display { |         let time_display = match flags.time_display { | ||||||
|             TimeDisplayType::Absolute => timezone |             TimeDisplayType::Absolute => { | ||||||
|                 .timestamp(self.utc_time.timestamp(), 0) |                 self.utc_time.with_timezone(timezone).format("%Y-%m-%d %H:%M:%S").to_string() | ||||||
|                 .format("%Y-%m-%d %H:%M:%S") |             } | ||||||
|                 .to_string(), |  | ||||||
|  |  | ||||||
|             TimeDisplayType::Relative => format!("<t:{}:R>", self.utc_time.timestamp()), |             TimeDisplayType::Relative => format!("<t:{}:R>", self.utc_time.timestamp()), | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|         if self.interval_seconds.is_some() || self.interval_months.is_some() { |         if self.interval_seconds.is_some() | ||||||
|  |             || self.interval_days.is_some() | ||||||
|  |             || self.interval_months.is_some() | ||||||
|  |         { | ||||||
|             format!( |             format!( | ||||||
|                 "'{}' *occurs next at* **{}**, repeating (set by {})", |                 "'{}' *occurs next at* **{}**, repeating (set by {})\n", | ||||||
|                 self.display_content(), |                 self.display_content(), | ||||||
|                 time_display, |                 time_display, | ||||||
|                 self.set_by.map(|i| format!("<@{}>", i)).unwrap_or_else(|| "unknown".to_string()) |                 self.set_by.map(|i| format!("<@{}>", i)).unwrap_or_else(|| "unknown".to_string()) | ||||||
|             ) |             ) | ||||||
|         } else { |         } else { | ||||||
|             format!( |             format!( | ||||||
|                 "'{}' *occurs next at* **{}** (set by {})", |                 "'{}' *occurs next at* **{}** (set by {})\n", | ||||||
|                 self.display_content(), |                 self.display_content(), | ||||||
|                 time_display, |                 time_display, | ||||||
|                 self.set_by.map(|i| format!("<@{}>", i)).unwrap_or_else(|| "unknown".to_string()) |                 self.set_by.map(|i| format!("<@{}>", i)).unwrap_or_else(|| "unknown".to_string()) | ||||||
|   | |||||||
| @@ -1,9 +1,9 @@ | |||||||
| use chrono::NaiveDateTime; | use chrono::{DateTime, Utc}; | ||||||
| use sqlx::MySqlPool; | use sqlx::MySqlPool; | ||||||
|  |  | ||||||
| pub struct Timer { | pub struct Timer { | ||||||
|     pub name: String, |     pub name: String, | ||||||
|     pub start_time: NaiveDateTime, |     pub start_time: DateTime<Utc>, | ||||||
|     pub owner: u64, |     pub owner: u64, | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
| use log::error; | use log::error; | ||||||
| use poise::serenity::{http::CacheHttp, model::id::UserId}; | use poise::serenity_prelude::{http::CacheHttp, model::id::UserId}; | ||||||
| use sqlx::MySqlPool; | use sqlx::MySqlPool; | ||||||
|  |  | ||||||
| use crate::consts::LOCAL_TIMEZONE; | use crate::consts::LOCAL_TIMEZONE; | ||||||
| @@ -10,6 +10,7 @@ pub struct UserData { | |||||||
|     pub user: u64, |     pub user: u64, | ||||||
|     pub dm_channel: u32, |     pub dm_channel: u32, | ||||||
|     pub timezone: String, |     pub timezone: String, | ||||||
|  |     pub allowed_dm: bool, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl UserData { | impl UserData { | ||||||
| @@ -21,7 +22,7 @@ impl UserData { | |||||||
|  |  | ||||||
|         match sqlx::query!( |         match sqlx::query!( | ||||||
|             " |             " | ||||||
| SELECT timezone FROM users WHERE user = ? | SELECT IFNULL(timezone, 'UTC') AS timezone FROM users WHERE user = ? | ||||||
|             ", |             ", | ||||||
|             user_id |             user_id | ||||||
|         ) |         ) | ||||||
| @@ -46,7 +47,7 @@ SELECT timezone FROM users WHERE user = ? | |||||||
|         match sqlx::query_as_unchecked!( |         match sqlx::query_as_unchecked!( | ||||||
|             Self, |             Self, | ||||||
|             " |             " | ||||||
| SELECT id, user, dm_channel, IF(timezone IS NULL, ?, timezone) AS timezone FROM users WHERE user = ? | SELECT id, user, dm_channel, IF(timezone IS NULL, ?, timezone) AS timezone, allowed_dm FROM users WHERE user = ? | ||||||
|             ", |             ", | ||||||
|             *LOCAL_TIMEZONE, |             *LOCAL_TIMEZONE, | ||||||
|             user_id.0 |             user_id.0 | ||||||
| @@ -71,7 +72,7 @@ INSERT IGNORE INTO channels (channel) VALUES (?) | |||||||
|  |  | ||||||
|                 sqlx::query!( |                 sqlx::query!( | ||||||
|                     " |                     " | ||||||
| INSERT INTO users (user, dm_channel, timezone) VALUES (?, (SELECT id FROM channels WHERE channel = ?), ?) | INSERT INTO users (name, user, dm_channel, timezone) VALUES ('', ?, (SELECT id FROM channels WHERE channel = ?), ?) | ||||||
|                     ", |                     ", | ||||||
|                     user_id.0, |                     user_id.0, | ||||||
|                     dm_channel.id.0, |                     dm_channel.id.0, | ||||||
| @@ -83,7 +84,7 @@ INSERT INTO users (user, dm_channel, timezone) VALUES (?, (SELECT id FROM channe | |||||||
|                 Ok(sqlx::query_as_unchecked!( |                 Ok(sqlx::query_as_unchecked!( | ||||||
|                     Self, |                     Self, | ||||||
|                     " |                     " | ||||||
| SELECT id, user, dm_channel, timezone FROM users WHERE user = ? | SELECT id, user, dm_channel, timezone, allowed_dm FROM users WHERE user = ? | ||||||
|                     ", |                     ", | ||||||
|                     user_id.0 |                     user_id.0 | ||||||
|                 ) |                 ) | ||||||
| @@ -102,9 +103,10 @@ SELECT id, user, dm_channel, timezone FROM users WHERE user = ? | |||||||
|     pub async fn commit_changes(&self, pool: &MySqlPool) { |     pub async fn commit_changes(&self, pool: &MySqlPool) { | ||||||
|         sqlx::query!( |         sqlx::query!( | ||||||
|             " |             " | ||||||
| UPDATE users SET timezone = ? WHERE id = ? | UPDATE users SET timezone = ?, allowed_dm = ? WHERE id = ? | ||||||
|             ", |             ", | ||||||
|             self.timezone, |             self.timezone, | ||||||
|  |             self.allowed_dm, | ||||||
|             self.id |             self.id | ||||||
|         ) |         ) | ||||||
|         .execute(pool) |         .execute(pool) | ||||||
|   | |||||||
| @@ -211,14 +211,12 @@ pub async fn natural_parser(time: &str, timezone: &str) -> Option<i64> { | |||||||
|         .output() |         .output() | ||||||
|         .await |         .await | ||||||
|         .ok() |         .ok() | ||||||
|         .map(|inner| { |         .and_then(|inner| { | ||||||
|             if inner.status.success() { |             if inner.status.success() { | ||||||
|                 Some(from_utf8(&*inner.stdout).unwrap().parse::<i64>().unwrap()) |                 Some(from_utf8(&*inner.stdout).unwrap().parse::<i64>().unwrap()) | ||||||
|             } else { |             } else { | ||||||
|                 None |                 None | ||||||
|             } |             } | ||||||
|         }) |         }) | ||||||
|         .flatten() |         .and_then(|inner| if inner < 0 { None } else { Some(inner) }) | ||||||
|         .map(|inner| if inner < 0 { None } else { Some(inner) }) |  | ||||||
|         .flatten() |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										15
									
								
								src/utils.rs
									
									
									
									
									
								
							
							
						
						| @@ -1,10 +1,11 @@ | |||||||
| use poise::{ | use poise::{ | ||||||
|     serenity::{ |     serenity_prelude as serenity, | ||||||
|  |     serenity_prelude::{ | ||||||
|         builder::CreateApplicationCommands, |         builder::CreateApplicationCommands, | ||||||
|         http::CacheHttp, |         http::CacheHttp, | ||||||
|  |         interaction::MessageFlags, | ||||||
|         model::id::{GuildId, UserId}, |         model::id::{GuildId, UserId}, | ||||||
|     }, |     }, | ||||||
|     serenity_prelude as serenity, |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
| @@ -13,10 +14,10 @@ use crate::{ | |||||||
| }; | }; | ||||||
|  |  | ||||||
| pub async fn register_application_commands( | pub async fn register_application_commands( | ||||||
|     ctx: &poise::serenity::client::Context, |     ctx: &serenity::Context, | ||||||
|     framework: &poise::Framework<Data, Error>, |     framework: &poise::Framework<Data, Error>, | ||||||
|     guild_id: Option<GuildId>, |     guild_id: Option<GuildId>, | ||||||
| ) -> Result<(), poise::serenity::Error> { | ) -> Result<(), serenity::Error> { | ||||||
|     let mut commands_builder = CreateApplicationCommands::default(); |     let mut commands_builder = CreateApplicationCommands::default(); | ||||||
|     let commands = &framework.options().commands; |     let commands = &framework.options().commands; | ||||||
|     for command in commands { |     for command in commands { | ||||||
| @@ -27,7 +28,7 @@ pub async fn register_application_commands( | |||||||
|             commands_builder.add_application_command(context_menu_command); |             commands_builder.add_application_command(context_menu_command); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     let commands_builder = poise::serenity::json::Value::Array(commands_builder.0); |     let commands_builder = poise::serenity_prelude::json::Value::Array(commands_builder.0); | ||||||
|  |  | ||||||
|     if let Some(guild_id) = guild_id { |     if let Some(guild_id) = guild_id { | ||||||
|         ctx.http.create_guild_application_commands(guild_id.0, &commands_builder).await?; |         ctx.http.create_guild_application_commands(guild_id.0, &commands_builder).await?; | ||||||
| @@ -82,7 +83,7 @@ pub fn send_as_initial_response( | |||||||
|         components, |         components, | ||||||
|         ephemeral, |         ephemeral, | ||||||
|         allowed_mentions, |         allowed_mentions, | ||||||
|         reference_message: _, // can't reply to a message in interactions |         reply: _, | ||||||
|     } = data; |     } = data; | ||||||
|  |  | ||||||
|     if let Some(content) = content { |     if let Some(content) = content { | ||||||
| @@ -102,6 +103,6 @@ pub fn send_as_initial_response( | |||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|     if ephemeral { |     if ephemeral { | ||||||
|         f.flags(serenity::InteractionApplicationCommandCallbackDataFlags::EPHEMERAL); |         f.flags(MessageFlags::EPHEMERAL); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								systemd/reminder-rs.service
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | |||||||
|  | [Unit] | ||||||
|  | Description=Reminder Bot | ||||||
|  |  | ||||||
|  | [Service] | ||||||
|  | User=reminder | ||||||
|  | Type=simple | ||||||
|  | ExecStart=/usr/bin/reminder-rs | ||||||
|  | WorkingDirectory=/etc/reminder-rs | ||||||
|  | Restart=always | ||||||
|  | RestartSec=4 | ||||||
|  | Environment="reminder_rs=warn,postman=warn" | ||||||
|  |  | ||||||
|  | [Install] | ||||||
|  | WantedBy=multi-user.target | ||||||
| @@ -1,21 +1,22 @@ | |||||||
| [package] | [package] | ||||||
| name = "reminder_web" | name = "reminder_web" | ||||||
| version = "0.1.0" | version = "0.1.3" | ||||||
| authors = ["jellywx <judesouthworth@pm.me>"] | authors = ["jellywx <judesouthworth@pm.me>"] | ||||||
| edition = "2018" | edition = "2018" | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| rocket = { git = "https://github.com/SergioBenitez/Rocket", branch = "master", features = ["tls", "secrets", "json"] } | rocket = { git = "https://github.com/SergioBenitez/Rocket", branch = "master", features = ["tls", "secrets", "json"] } | ||||||
| rocket_dyn_templates = { git = "https://github.com/SergioBenitez/Rocket", branch = "master", features = ["tera"] } | rocket_dyn_templates = { git = "https://github.com/SergioBenitez/Rocket", branch = "master", features = ["tera"] } | ||||||
| serenity = { version = "0.11.1", default-features = false, features = ["builder", "cache", "client", "gateway", "http", "model", "utils", "rustls_backend"] } | serenity = { version = "0.11", default-features = false, features = ["builder", "cache", "client", "gateway", "http", "model", "utils", "rustls_backend"] } | ||||||
| oauth2 = "4" | oauth2 = "4" | ||||||
| log = "0.4" | log = "0.4" | ||||||
| reqwest = "0.11" | reqwest = "0.11" | ||||||
| serde = { version = "1.0", features = ["derive"] } | serde = { version = "1.0", features = ["derive"] } | ||||||
| serde_json = "1.0" | sqlx = { version = "0.7", features = ["runtime-tokio-rustls", "macros", "mysql", "chrono", "json"] } | ||||||
| sqlx = { version = "0.5", features = ["runtime-tokio-rustls", "macros", "mysql", "chrono", "json"] } |  | ||||||
| chrono = "0.4" | chrono = "0.4" | ||||||
| chrono-tz = "0.5" | chrono-tz = "0.8" | ||||||
| lazy_static = "1.4.0" | lazy_static = "1.4.0" | ||||||
| rand = "0.7" | rand = "0.8" | ||||||
| base64 = "0.13" | base64 = "0.13" | ||||||
|  | csv = "1.2" | ||||||
|  | prometheus = "0.13.3" | ||||||
|   | |||||||
							
								
								
									
										40
									
								
								web/src/catchers.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,40 @@ | |||||||
|  | use std::collections::HashMap; | ||||||
|  |  | ||||||
|  | use rocket::serde::json::json; | ||||||
|  | use rocket_dyn_templates::Template; | ||||||
|  |  | ||||||
|  | use crate::JsonValue; | ||||||
|  |  | ||||||
|  | #[catch(403)] | ||||||
|  | pub(crate) async fn forbidden() -> Template { | ||||||
|  |     let map: HashMap<String, String> = HashMap::new(); | ||||||
|  |     Template::render("errors/403", &map) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[catch(500)] | ||||||
|  | pub(crate) async fn internal_server_error() -> Template { | ||||||
|  |     let map: HashMap<String, String> = HashMap::new(); | ||||||
|  |     Template::render("errors/500", &map) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[catch(401)] | ||||||
|  | pub(crate) async fn not_authorized() -> Template { | ||||||
|  |     let map: HashMap<String, String> = HashMap::new(); | ||||||
|  |     Template::render("errors/401", &map) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[catch(404)] | ||||||
|  | pub(crate) async fn not_found() -> Template { | ||||||
|  |     let map: HashMap<String, String> = HashMap::new(); | ||||||
|  |     Template::render("errors/404", &map) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[catch(413)] | ||||||
|  | pub(crate) async fn payload_too_large() -> JsonValue { | ||||||
|  |     json!({"error": "Data too large.", "errors": ["Data too large."]}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[catch(422)] | ||||||
|  | pub(crate) async fn unprocessable_entity() -> JsonValue { | ||||||
|  |     json!({"error": "Invalid request.", "errors": ["Invalid request."]}) | ||||||
|  | } | ||||||
| @@ -2,6 +2,7 @@ pub const DISCORD_OAUTH_TOKEN: &'static str = "https://discord.com/api/oauth2/to | |||||||
| pub const DISCORD_OAUTH_AUTHORIZE: &'static str = "https://discord.com/api/oauth2/authorize"; | pub const DISCORD_OAUTH_AUTHORIZE: &'static str = "https://discord.com/api/oauth2/authorize"; | ||||||
| pub const DISCORD_API: &'static str = "https://discord.com/api"; | pub const DISCORD_API: &'static str = "https://discord.com/api"; | ||||||
|  |  | ||||||
|  | pub const MAX_NAME_LENGTH: usize = 100; | ||||||
| pub const MAX_CONTENT_LENGTH: usize = 2000; | pub const MAX_CONTENT_LENGTH: usize = 2000; | ||||||
| pub const MAX_EMBED_DESCRIPTION_LENGTH: usize = 4096; | pub const MAX_EMBED_DESCRIPTION_LENGTH: usize = 4096; | ||||||
| pub const MAX_EMBED_TITLE_LENGTH: usize = 256; | pub const MAX_EMBED_TITLE_LENGTH: usize = 256; | ||||||
| @@ -26,16 +27,12 @@ use serenity::model::prelude::AttachmentType; | |||||||
|  |  | ||||||
| lazy_static! { | lazy_static! { | ||||||
|     pub static ref DEFAULT_AVATAR: AttachmentType<'static> = ( |     pub static ref DEFAULT_AVATAR: AttachmentType<'static> = ( | ||||||
|         include_bytes!(concat!( |         include_bytes!(concat!(env!("CARGO_MANIFEST_DIR"), "/../assets/webhook.jpg")) as &[u8], | ||||||
|             env!("CARGO_MANIFEST_DIR"), |         "webhook.jpg", | ||||||
|             "/../assets/", |  | ||||||
|             env!("WEBHOOK_AVATAR", "WEBHOOK_AVATAR not provided for compilation") |  | ||||||
|         )) as &[u8], |  | ||||||
|         env!("WEBHOOK_AVATAR"), |  | ||||||
|     ) |     ) | ||||||
|         .into(); |         .into(); | ||||||
|     pub static ref SUBSCRIPTION_ROLES: HashSet<u64> = HashSet::from_iter( |     pub static ref SUBSCRIPTION_ROLES: HashSet<u64> = HashSet::from_iter( | ||||||
|         env::var("SUBSCRIPTION_ROLES") |         env::var("PATREON_ROLE_ID") | ||||||
|             .map(|var| var |             .map(|var| var | ||||||
|                 .split(',') |                 .split(',') | ||||||
|                 .filter_map(|item| { item.parse::<u64>().ok() }) |                 .filter_map(|item| { item.parse::<u64>().ok() }) | ||||||
| @@ -43,7 +40,7 @@ lazy_static! { | |||||||
|             .unwrap_or_else(|_| Vec::new()) |             .unwrap_or_else(|_| Vec::new()) | ||||||
|     ); |     ); | ||||||
|     pub static ref CNC_GUILD: Option<u64> = |     pub static ref CNC_GUILD: Option<u64> = | ||||||
|         env::var("CNC_GUILD").map(|var| var.parse::<u64>().ok()).ok().flatten(); |         env::var("PATREON_GUILD_ID").map(|var| var.parse::<u64>().ok()).ok().flatten(); | ||||||
|     pub static ref MIN_INTERVAL: u32 = env::var("MIN_INTERVAL") |     pub static ref MIN_INTERVAL: u32 = env::var("MIN_INTERVAL") | ||||||
|         .ok() |         .ok() | ||||||
|         .map(|inner| inner.parse::<u32>().ok()) |         .map(|inner| inner.parse::<u32>().ok()) | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								web/src/guards/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | pub(crate) mod transaction; | ||||||
							
								
								
									
										44
									
								
								web/src/guards/transaction.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,44 @@ | |||||||
|  | use rocket::{ | ||||||
|  |     http::Status, | ||||||
|  |     request::{FromRequest, Outcome}, | ||||||
|  |     Request, State, | ||||||
|  | }; | ||||||
|  | use sqlx::Pool; | ||||||
|  |  | ||||||
|  | use crate::Database; | ||||||
|  |  | ||||||
|  | pub struct Transaction<'a>(sqlx::Transaction<'a, Database>); | ||||||
|  |  | ||||||
|  | impl Transaction<'_> { | ||||||
|  |     pub fn executor(&mut self) -> impl sqlx::Executor<'_, Database = Database> { | ||||||
|  |         &mut *(self.0) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub async fn commit(self) -> Result<(), sqlx::Error> { | ||||||
|  |         self.0.commit().await | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Debug)] | ||||||
|  | pub enum TransactionError { | ||||||
|  |     Error(sqlx::Error), | ||||||
|  |     Missing, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[rocket::async_trait] | ||||||
|  | impl<'r> FromRequest<'r> for Transaction<'r> { | ||||||
|  |     type Error = TransactionError; | ||||||
|  |  | ||||||
|  |     async fn from_request(request: &'r Request<'_>) -> Outcome<Self, Self::Error> { | ||||||
|  |         match request.guard::<&State<Pool<Database>>>().await { | ||||||
|  |             Outcome::Success(pool) => match pool.begin().await { | ||||||
|  |                 Ok(transaction) => Outcome::Success(Transaction(transaction)), | ||||||
|  |                 Err(e) => { | ||||||
|  |                     Outcome::Failure((Status::InternalServerError, TransactionError::Error(e))) | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             Outcome::Failure(e) => Outcome::Failure((e.0, TransactionError::Missing)), | ||||||
|  |             Outcome::Forward(f) => Outcome::Forward(f), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										193
									
								
								web/src/lib.rs
									
									
									
									
									
								
							
							
						
						| @@ -4,13 +4,17 @@ extern crate rocket; | |||||||
| mod consts; | mod consts; | ||||||
| #[macro_use] | #[macro_use] | ||||||
| mod macros; | mod macros; | ||||||
|  | mod catchers; | ||||||
|  | mod guards; | ||||||
|  | mod metrics; | ||||||
| mod routes; | mod routes; | ||||||
|  |  | ||||||
| use std::{collections::HashMap, env}; | use std::{env, path::Path}; | ||||||
|  |  | ||||||
| use oauth2::{basic::BasicClient, AuthUrl, ClientId, ClientSecret, RedirectUrl, TokenUrl}; | use oauth2::{basic::BasicClient, AuthUrl, ClientId, ClientSecret, RedirectUrl, TokenUrl}; | ||||||
| use rocket::{ | use rocket::{ | ||||||
|     fs::FileServer, |     fs::FileServer, | ||||||
|  |     http::CookieJar, | ||||||
|     serde::json::{json, Value as JsonValue}, |     serde::json::{json, Value as JsonValue}, | ||||||
|     tokio::sync::broadcast::Sender, |     tokio::sync::broadcast::Sender, | ||||||
| }; | }; | ||||||
| @@ -22,7 +26,10 @@ use serenity::{ | |||||||
| }; | }; | ||||||
| use sqlx::{MySql, Pool}; | use sqlx::{MySql, Pool}; | ||||||
|  |  | ||||||
| use crate::consts::{CNC_GUILD, DISCORD_OAUTH_AUTHORIZE, DISCORD_OAUTH_TOKEN, SUBSCRIPTION_ROLES}; | use crate::{ | ||||||
|  |     consts::{CNC_GUILD, DISCORD_OAUTH_AUTHORIZE, DISCORD_OAUTH_TOKEN, SUBSCRIPTION_ROLES}, | ||||||
|  |     metrics::{init_metrics, MetricProducer}, | ||||||
|  | }; | ||||||
|  |  | ||||||
| type Database = MySql; | type Database = MySql; | ||||||
|  |  | ||||||
| @@ -32,50 +39,20 @@ enum Error { | |||||||
|     Serenity(serenity::Error), |     Serenity(serenity::Error), | ||||||
| } | } | ||||||
|  |  | ||||||
| #[catch(401)] |  | ||||||
| async fn not_authorized() -> Template { |  | ||||||
|     let map: HashMap<String, String> = HashMap::new(); |  | ||||||
|     Template::render("errors/401", &map) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[catch(403)] |  | ||||||
| async fn forbidden() -> Template { |  | ||||||
|     let map: HashMap<String, String> = HashMap::new(); |  | ||||||
|     Template::render("errors/403", &map) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[catch(404)] |  | ||||||
| async fn not_found() -> Template { |  | ||||||
|     let map: HashMap<String, String> = HashMap::new(); |  | ||||||
|     Template::render("errors/404", &map) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[catch(413)] |  | ||||||
| async fn payload_too_large() -> JsonValue { |  | ||||||
|     json!({"error": "Data too large.", "errors": ["Data too large."]}) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[catch(422)] |  | ||||||
| async fn unprocessable_entity() -> JsonValue { |  | ||||||
|     json!({"error": "Invalid request.", "errors": ["Invalid request."]}) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[catch(500)] |  | ||||||
| async fn internal_server_error() -> Template { |  | ||||||
|     let map: HashMap<String, String> = HashMap::new(); |  | ||||||
|     Template::render("errors/500", &map) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| pub async fn initialize( | pub async fn initialize( | ||||||
|     kill_channel: Sender<()>, |     kill_channel: Sender<()>, | ||||||
|     serenity_context: Context, |     serenity_context: Context, | ||||||
|     db_pool: Pool<Database>, |     db_pool: Pool<Database>, | ||||||
| ) -> Result<(), Box<dyn std::error::Error>> { | ) -> Result<(), Box<dyn std::error::Error>> { | ||||||
|     info!("Checking environment variables..."); |     info!("Checking environment variables..."); | ||||||
|     env::var("OAUTH2_CLIENT_ID").expect("`OAUTH2_CLIENT_ID' not supplied"); |  | ||||||
|     env::var("OAUTH2_CLIENT_SECRET").expect("`OAUTH2_CLIENT_SECRET' not supplied"); |     if env::var("OFFLINE").map_or(true, |v| v != "1") { | ||||||
|     env::var("OAUTH2_DISCORD_CALLBACK").expect("`OAUTH2_DISCORD_CALLBACK' not supplied"); |         env::var("OAUTH2_CLIENT_ID").expect("`OAUTH2_CLIENT_ID' not supplied"); | ||||||
|     env::var("PATREON_GUILD_ID").expect("`PATREON_GUILD' not supplied"); |         env::var("OAUTH2_CLIENT_SECRET").expect("`OAUTH2_CLIENT_SECRET' not supplied"); | ||||||
|  |         env::var("OAUTH2_DISCORD_CALLBACK").expect("`OAUTH2_DISCORD_CALLBACK' not supplied"); | ||||||
|  |         env::var("PATREON_GUILD_ID").expect("`PATREON_GUILD_ID' not supplied"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     info!("Done!"); |     info!("Done!"); | ||||||
|  |  | ||||||
|     let oauth2_client = BasicClient::new( |     let oauth2_client = BasicClient::new( | ||||||
| @@ -88,32 +65,40 @@ pub async fn initialize( | |||||||
|  |  | ||||||
|     let reqwest_client = reqwest::Client::new(); |     let reqwest_client = reqwest::Client::new(); | ||||||
|  |  | ||||||
|  |     let static_path = | ||||||
|  |         if Path::new("web/static").exists() { "web/static" } else { "/lib/reminder-rs/static" }; | ||||||
|  |  | ||||||
|  |     init_metrics(); | ||||||
|  |  | ||||||
|     rocket::build() |     rocket::build() | ||||||
|  |         .attach(MetricProducer) | ||||||
|         .attach(Template::fairing()) |         .attach(Template::fairing()) | ||||||
|         .register( |         .register( | ||||||
|             "/", |             "/", | ||||||
|             catchers![ |             catchers![ | ||||||
|                 not_authorized, |                 catchers::not_authorized, | ||||||
|                 forbidden, |                 catchers::forbidden, | ||||||
|                 not_found, |                 catchers::not_found, | ||||||
|                 internal_server_error, |                 catchers::internal_server_error, | ||||||
|                 unprocessable_entity, |                 catchers::unprocessable_entity, | ||||||
|                 payload_too_large, |                 catchers::payload_too_large, | ||||||
|             ], |             ], | ||||||
|         ) |         ) | ||||||
|         .manage(oauth2_client) |         .manage(oauth2_client) | ||||||
|         .manage(reqwest_client) |         .manage(reqwest_client) | ||||||
|         .manage(serenity_context) |         .manage(serenity_context) | ||||||
|         .manage(db_pool) |         .manage(db_pool) | ||||||
|         .mount("/static", FileServer::from(concat!(env!("CARGO_MANIFEST_DIR"), "/static"))) |         .mount("/static", FileServer::from(static_path)) | ||||||
|         .mount( |         .mount( | ||||||
|             "/", |             "/", | ||||||
|             routes![ |             routes![ | ||||||
|                 routes::index, |  | ||||||
|                 routes::cookies, |                 routes::cookies, | ||||||
|  |                 routes::index, | ||||||
|  |                 routes::metrics::metrics, | ||||||
|                 routes::privacy, |                 routes::privacy, | ||||||
|  |                 routes::report::report_error, | ||||||
|  |                 routes::return_to_same_site, | ||||||
|                 routes::terms, |                 routes::terms, | ||||||
|                 routes::return_to_same_site |  | ||||||
|             ], |             ], | ||||||
|         ) |         ) | ||||||
|         .mount( |         .mount( | ||||||
| @@ -126,29 +111,45 @@ pub async fn initialize( | |||||||
|                 routes::help_timers, |                 routes::help_timers, | ||||||
|                 routes::help_todo_lists, |                 routes::help_todo_lists, | ||||||
|                 routes::help_macros, |                 routes::help_macros, | ||||||
|  |                 routes::help_intervals, | ||||||
|  |                 routes::help_dashboard, | ||||||
|  |                 routes::help_iemanager, | ||||||
|  |             ], | ||||||
|  |         ) | ||||||
|  |         .mount( | ||||||
|  |             "/login", | ||||||
|  |             routes![ | ||||||
|  |                 routes::login::discord_login, | ||||||
|  |                 routes::login::discord_logout, | ||||||
|  |                 routes::login::discord_callback | ||||||
|             ], |             ], | ||||||
|         ) |         ) | ||||||
|         .mount("/login", routes![routes::login::discord_login, routes::login::discord_callback]) |  | ||||||
|         .mount( |         .mount( | ||||||
|             "/dashboard", |             "/dashboard", | ||||||
|             routes![ |             routes![ | ||||||
|                 routes::dashboard::dashboard, |                 routes::dashboard::dashboard, | ||||||
|                 routes::dashboard::dashboard_home, |                 routes::dashboard::dashboard_home, | ||||||
|                 routes::dashboard::user::get_user_info, |                 routes::dashboard::api::user::get_user_info, | ||||||
|                 routes::dashboard::user::update_user_info, |                 routes::dashboard::api::user::update_user_info, | ||||||
|                 routes::dashboard::user::get_user_guilds, |                 routes::dashboard::api::user::get_user_guilds, | ||||||
|                 routes::dashboard::guild::get_guild_patreon, |                 routes::dashboard::api::guild::get_guild_info, | ||||||
|                 routes::dashboard::guild::get_guild_channels, |                 routes::dashboard::api::guild::get_guild_channels, | ||||||
|                 routes::dashboard::guild::get_guild_roles, |                 routes::dashboard::api::guild::get_guild_roles, | ||||||
|                 routes::dashboard::guild::get_reminder_templates, |                 routes::dashboard::api::guild::get_reminder_templates, | ||||||
|                 routes::dashboard::guild::create_reminder_template, |                 routes::dashboard::api::guild::create_reminder_template, | ||||||
|                 routes::dashboard::guild::delete_reminder_template, |                 routes::dashboard::api::guild::delete_reminder_template, | ||||||
|                 routes::dashboard::guild::create_reminder, |                 routes::dashboard::api::guild::create_guild_reminder, | ||||||
|                 routes::dashboard::guild::get_reminders, |                 routes::dashboard::api::guild::get_reminders, | ||||||
|                 routes::dashboard::guild::edit_reminder, |                 routes::dashboard::api::guild::edit_reminder, | ||||||
|                 routes::dashboard::guild::delete_reminder, |                 routes::dashboard::api::guild::delete_reminder, | ||||||
|  |                 routes::dashboard::export::export_reminders, | ||||||
|  |                 routes::dashboard::export::export_reminder_templates, | ||||||
|  |                 routes::dashboard::export::export_todos, | ||||||
|  |                 routes::dashboard::export::import_reminders, | ||||||
|  |                 routes::dashboard::export::import_todos, | ||||||
|             ], |             ], | ||||||
|         ) |         ) | ||||||
|  |         .mount("/admin", routes![routes::admin::admin_dashboard_home, routes::admin::bot_data]) | ||||||
|         .launch() |         .launch() | ||||||
|         .await?; |         .await?; | ||||||
|  |  | ||||||
| @@ -165,6 +166,8 @@ pub async fn initialize( | |||||||
| } | } | ||||||
|  |  | ||||||
| pub async fn check_subscription(cache_http: impl CacheHttp, user_id: impl Into<UserId>) -> bool { | pub async fn check_subscription(cache_http: impl CacheHttp, user_id: impl Into<UserId>) -> bool { | ||||||
|  |     offline!(true); | ||||||
|  |  | ||||||
|     if let Some(subscription_guild) = *CNC_GUILD { |     if let Some(subscription_guild) = *CNC_GUILD { | ||||||
|         let guild_member = GuildId(subscription_guild).member(cache_http, user_id).await; |         let guild_member = GuildId(subscription_guild).member(cache_http, user_id).await; | ||||||
|  |  | ||||||
| @@ -186,6 +189,8 @@ pub async fn check_guild_subscription( | |||||||
|     cache_http: impl CacheHttp, |     cache_http: impl CacheHttp, | ||||||
|     guild_id: impl Into<GuildId>, |     guild_id: impl Into<GuildId>, | ||||||
| ) -> bool { | ) -> bool { | ||||||
|  |     offline!(true); | ||||||
|  |  | ||||||
|     if let Some(guild) = cache_http.cache().unwrap().guild(guild_id) { |     if let Some(guild) = cache_http.cache().unwrap().guild(guild_id) { | ||||||
|         let owner = guild.owner_id; |         let owner = guild.owner_id; | ||||||
|  |  | ||||||
| @@ -194,3 +199,65 @@ pub async fn check_guild_subscription( | |||||||
|         false |         false | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | pub async fn check_authorization( | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     ctx: &Context, | ||||||
|  |     guild: u64, | ||||||
|  | ) -> Result<(), JsonValue> { | ||||||
|  |     let user_id = cookies.get_private("userid").map(|c| c.value().parse::<u64>().ok()).flatten(); | ||||||
|  |  | ||||||
|  |     if std::env::var("OFFLINE").map_or(true, |v| v != "1") { | ||||||
|  |         match user_id { | ||||||
|  |             Some(user_id) => { | ||||||
|  |                 let admin_id = std::env::var("ADMIN_ID") | ||||||
|  |                     .map_or(false, |u| u.parse::<u64>().map_or(false, |u| u == user_id)); | ||||||
|  |  | ||||||
|  |                 if admin_id { | ||||||
|  |                     return Ok(()); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 match GuildId(guild).to_guild_cached(ctx) { | ||||||
|  |                     Some(guild) => { | ||||||
|  |                         let member_res = guild.member(ctx, UserId(user_id)).await; | ||||||
|  |  | ||||||
|  |                         match member_res { | ||||||
|  |                             Err(_) => { | ||||||
|  |                                 return Err(json!({"error": "User not in guild"})); | ||||||
|  |                             } | ||||||
|  |  | ||||||
|  |                             Ok(member) => { | ||||||
|  |                                 let permissions_res = member.permissions(ctx); | ||||||
|  |  | ||||||
|  |                                 match permissions_res { | ||||||
|  |                                     Err(_) => { | ||||||
|  |                                         return Err(json!({"error": "Couldn't fetch permissions"})); | ||||||
|  |                                     } | ||||||
|  |  | ||||||
|  |                                     Ok(permissions) => { | ||||||
|  |                                         if !(permissions.manage_messages() | ||||||
|  |                                             || permissions.manage_guild() | ||||||
|  |                                             || permissions.administrator()) | ||||||
|  |                                         { | ||||||
|  |                                             return Err(json!({"error": "Incorrect permissions"})); | ||||||
|  |                                         } | ||||||
|  |                                     } | ||||||
|  |                                 } | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     None => { | ||||||
|  |                         return Err(json!({"error": "Bot not in guild"})); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             None => { | ||||||
|  |                 return Err(json!({"error": "User not authorized"})); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|   | |||||||
| @@ -1,7 +1,15 @@ | |||||||
|  | macro_rules! offline { | ||||||
|  |     ($field:expr) => { | ||||||
|  |         if std::env::var("OFFLINE").map_or(false, |v| v == "1") { | ||||||
|  |             return $field; | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  | } | ||||||
|  |  | ||||||
| macro_rules! check_length { | macro_rules! check_length { | ||||||
|     ($max:ident, $field:expr) => { |     ($max:ident, $field:expr) => { | ||||||
|         if $field.len() > $max { |         if $field.len() > $max { | ||||||
|             return json!({ "error": format!("{} exceeded", stringify!($max)) }); |             return Err(json!({ "error": format!("{} exceeded", stringify!($max)) })); | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
|     ($max:ident, $field:expr, $($fields:expr),+) => { |     ($max:ident, $field:expr, $($fields:expr),+) => { | ||||||
| @@ -25,7 +33,7 @@ macro_rules! check_length_opt { | |||||||
| macro_rules! check_url { | macro_rules! check_url { | ||||||
|     ($field:expr) => { |     ($field:expr) => { | ||||||
|         if !($field.starts_with("http://") || $field.starts_with("https://")) { |         if !($field.starts_with("http://") || $field.starts_with("https://")) { | ||||||
|             return json!({ "error": "URL invalid" }); |             return Err(json!({ "error": "URL invalid" })); | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
|     ($field:expr, $($fields:expr),+) => { |     ($field:expr, $($fields:expr),+) => { | ||||||
| @@ -46,40 +54,6 @@ macro_rules! check_url_opt { | |||||||
|     }; |     }; | ||||||
| } | } | ||||||
|  |  | ||||||
| macro_rules! check_authorization { |  | ||||||
|     ($cookies:expr, $ctx:expr, $guild:expr) => { |  | ||||||
|         use serenity::model::id::UserId; |  | ||||||
|  |  | ||||||
|         let user_id = $cookies.get_private("userid").map(|c| c.value().parse::<u64>().ok()).flatten(); |  | ||||||
|  |  | ||||||
|         match user_id { |  | ||||||
|             Some(user_id) => { |  | ||||||
|                 match GuildId($guild).to_guild_cached($ctx) { |  | ||||||
|                     Some(guild) => { |  | ||||||
|                         let member = guild.member($ctx, UserId(user_id)).await; |  | ||||||
|  |  | ||||||
|                         match member { |  | ||||||
|                             Err(_) => { |  | ||||||
|                                 return json!({"error": "User not in guild"}) |  | ||||||
|                             } |  | ||||||
|  |  | ||||||
|                             Ok(_) => {} |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     None => { |  | ||||||
|                         return json!({"error": "Bot not in guild"}) |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             None => { |  | ||||||
|                 return json!({"error": "User not authorized"}); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| macro_rules! update_field { | macro_rules! update_field { | ||||||
|     ($pool:expr, $error:ident, $reminder:ident.[$field:ident]) => { |     ($pool:expr, $error:ident, $reminder:ident.[$field:ident]) => { | ||||||
|         if let Some(value) = &$reminder.$field { |         if let Some(value) = &$reminder.$field { | ||||||
| @@ -117,3 +91,9 @@ macro_rules! update_field { | |||||||
|         update_field!($pool, $error, $reminder.[$($fields),+]); |         update_field!($pool, $error, $reminder.[$($fields),+]); | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | macro_rules! json_err { | ||||||
|  |     ($message:expr) => { | ||||||
|  |         Err(json!({ "error": $message })) | ||||||
|  |     }; | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										43
									
								
								web/src/metrics.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,43 @@ | |||||||
|  | use lazy_static::lazy_static; | ||||||
|  | use prometheus::{IntCounterVec, Opts, Registry}; | ||||||
|  | use rocket::{ | ||||||
|  |     fairing::{Fairing, Info, Kind}, | ||||||
|  |     Data, Request, Response, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | lazy_static! { | ||||||
|  |     pub static ref REGISTRY: Registry = Registry::new(); | ||||||
|  |     static ref REQUEST_COUNTER: IntCounterVec = | ||||||
|  |         IntCounterVec::new(Opts::new("requests", "Requests"), &["method", "route"]).unwrap(); | ||||||
|  |     static ref RESPONSE_COUNTER: IntCounterVec = | ||||||
|  |         IntCounterVec::new(Opts::new("responses", "Responses"), &["status", "route"]).unwrap(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | pub fn init_metrics() { | ||||||
|  |     REGISTRY.register(Box::new(REQUEST_COUNTER.clone())).unwrap(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | pub struct MetricProducer; | ||||||
|  |  | ||||||
|  | #[rocket::async_trait] | ||||||
|  | impl Fairing for MetricProducer { | ||||||
|  |     fn info(&self) -> Info { | ||||||
|  |         Info { name: "Metrics fairing", kind: Kind::Request } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async fn on_request(&self, req: &mut Request<'_>, _data: &mut Data<'_>) { | ||||||
|  |         if let Some(route) = req.route() { | ||||||
|  |             REQUEST_COUNTER | ||||||
|  |                 .with_label_values(&[req.method().as_str(), &route.uri.to_string()]) | ||||||
|  |                 .inc(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async fn on_response<'r>(&self, req: &'r Request<'_>, resp: &mut Response<'r>) { | ||||||
|  |         if let Some(route) = req.route() { | ||||||
|  |             RESPONSE_COUNTER | ||||||
|  |                 .with_label_values(&[&resp.status().code.to_string(), &route.uri.to_string()]) | ||||||
|  |                 .inc(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										218
									
								
								web/src/routes/admin.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,218 @@ | |||||||
|  | use std::{collections::HashMap, env}; | ||||||
|  |  | ||||||
|  | use chrono::{DateTime, Utc}; | ||||||
|  | use rocket::{ | ||||||
|  |     http::{CookieJar, Status}, | ||||||
|  |     serde::json::json, | ||||||
|  |     State, | ||||||
|  | }; | ||||||
|  | use rocket_dyn_templates::Template; | ||||||
|  | use serde::Serialize; | ||||||
|  | use sqlx::{MySql, Pool}; | ||||||
|  |  | ||||||
|  | use crate::routes::JsonResult; | ||||||
|  |  | ||||||
|  | fn is_admin(cookies: &CookieJar<'_>) -> bool { | ||||||
|  |     cookies | ||||||
|  |         .get_private("userid") | ||||||
|  |         .map_or(false, |cookie| Some(cookie.value().to_string()) == env::var("ADMIN_ID").ok()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[get("/")] | ||||||
|  | pub async fn admin_dashboard_home(cookies: &CookieJar<'_>) -> Result<Template, Status> { | ||||||
|  |     if let Some(cookie) = cookies.get_private("userid") { | ||||||
|  |         let map: HashMap<&str, String> = HashMap::new(); | ||||||
|  |         if Some(cookie.value().to_string()) == env::var("ADMIN_ID").ok() { | ||||||
|  |             Ok(Template::render("admin_dashboard", &map)) | ||||||
|  |         } else { | ||||||
|  |             Err(Status::Forbidden) | ||||||
|  |         } | ||||||
|  |     } else { | ||||||
|  |         Err(Status::Unauthorized) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Serialize)] | ||||||
|  | struct TimeFrame { | ||||||
|  |     time_key: DateTime<Utc>, | ||||||
|  |     count: i64, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[get("/data")] | ||||||
|  | pub async fn bot_data(cookies: &CookieJar<'_>, pool: &State<Pool<MySql>>) -> JsonResult { | ||||||
|  |     if !is_admin(cookies) { | ||||||
|  |         return json_err!("Not authorized"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let backlog = sqlx::query!( | ||||||
|  |         "SELECT COUNT(1) AS backlog FROM reminders WHERE `utc_time` < NOW() AND enabled = 1 AND `status` = 'pending'" | ||||||
|  |     ) | ||||||
|  |     .fetch_one(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     .unwrap(); | ||||||
|  |  | ||||||
|  |     let schedule_once = sqlx::query_as_unchecked!( | ||||||
|  |         TimeFrame, | ||||||
|  |         "SELECT | ||||||
|  |             FROM_UNIXTIME(FLOOR(UNIX_TIMESTAMP(`utc_time`) / 300) * 300) AS `time_key`, | ||||||
|  |             COUNT(1) AS `count` | ||||||
|  |         FROM reminders | ||||||
|  |         WHERE | ||||||
|  |             `utc_time` < DATE_ADD(NOW(), INTERVAL 1 DAY) AND | ||||||
|  |             `utc_time` >= NOW() AND | ||||||
|  |             `enabled` = 1 AND | ||||||
|  |             `status` = 'pending' AND | ||||||
|  |             `interval_seconds` IS NULL AND | ||||||
|  |             `interval_months` IS NULL AND | ||||||
|  |             `interval_days` IS NULL | ||||||
|  |         GROUP BY `time_key` | ||||||
|  |         ORDER BY `time_key`" | ||||||
|  |     ) | ||||||
|  |     .fetch_all(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     .unwrap(); | ||||||
|  |  | ||||||
|  |     let schedule_interval = sqlx::query_as_unchecked!( | ||||||
|  |         TimeFrame, | ||||||
|  |         "SELECT | ||||||
|  |             FROM_UNIXTIME(FLOOR(UNIX_TIMESTAMP(`utc_time`) / 300) * 300) AS `time_key`, | ||||||
|  |             COUNT(1) AS `count` | ||||||
|  |         FROM reminders | ||||||
|  |         WHERE | ||||||
|  |             `utc_time` < DATE_ADD(NOW(), INTERVAL 1 DAY) AND | ||||||
|  |             `utc_time` >= NOW() AND | ||||||
|  |             `status` = 'pending' AND | ||||||
|  |             `enabled` = 1 AND ( | ||||||
|  |                 `interval_seconds` IS NOT NULL OR | ||||||
|  |                 `interval_months` IS NOT NULL OR | ||||||
|  |                 `interval_days` IS NOT NULL | ||||||
|  |             ) | ||||||
|  |         GROUP BY `time_key` | ||||||
|  |         ORDER BY `time_key`" | ||||||
|  |     ) | ||||||
|  |     .fetch_all(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     .unwrap(); | ||||||
|  |  | ||||||
|  |     let schedule_once_long = sqlx::query_as_unchecked!( | ||||||
|  |         TimeFrame, | ||||||
|  |         "SELECT | ||||||
|  |             FROM_UNIXTIME(FLOOR(UNIX_TIMESTAMP(`utc_time`) / 86400) * 86400) AS `time_key`, | ||||||
|  |             COUNT(1) AS `count` | ||||||
|  |         FROM reminders | ||||||
|  |         WHERE | ||||||
|  |             `utc_time` < DATE_ADD(NOW(), INTERVAL 31 DAY) AND | ||||||
|  |             `utc_time` >= NOW() AND | ||||||
|  |             `enabled` = 1 AND | ||||||
|  |             `status` = 'pending' AND | ||||||
|  |             `interval_seconds` IS NULL AND | ||||||
|  |             `interval_months` IS NULL AND | ||||||
|  |             `interval_days` IS NULL | ||||||
|  |         GROUP BY `time_key` | ||||||
|  |         ORDER BY `time_key`" | ||||||
|  |     ) | ||||||
|  |     .fetch_all(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     .unwrap(); | ||||||
|  |  | ||||||
|  |     let schedule_interval_long = sqlx::query_as_unchecked!( | ||||||
|  |         TimeFrame, | ||||||
|  |         "SELECT | ||||||
|  |             FROM_UNIXTIME(FLOOR(UNIX_TIMESTAMP(`utc_time`) / 86400) * 86400) AS `time_key`, | ||||||
|  |             COUNT(1) AS `count` | ||||||
|  |         FROM reminders | ||||||
|  |         WHERE | ||||||
|  |             `utc_time` < DATE_ADD(NOW(), INTERVAL 31 DAY) AND | ||||||
|  |             `utc_time` >= NOW() AND | ||||||
|  |             `status` = 'pending' AND | ||||||
|  |             `enabled` = 1 AND ( | ||||||
|  |                 `interval_seconds` IS NOT NULL OR | ||||||
|  |                 `interval_months` IS NOT NULL OR | ||||||
|  |                 `interval_days` IS NOT NULL | ||||||
|  |             ) | ||||||
|  |         GROUP BY `time_key` | ||||||
|  |         ORDER BY `time_key`" | ||||||
|  |     ) | ||||||
|  |     .fetch_all(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     .unwrap(); | ||||||
|  |  | ||||||
|  |     let history = sqlx::query_as_unchecked!( | ||||||
|  |         TimeFrame, | ||||||
|  |         "SELECT | ||||||
|  |             FROM_UNIXTIME(FLOOR(UNIX_TIMESTAMP(`utc_time`) / 86400) * 86400) AS `time_key`, | ||||||
|  |             COUNT(1) AS `count` | ||||||
|  |         FROM stat | ||||||
|  |         WHERE | ||||||
|  |             `utc_time` > DATE_SUB(NOW(), INTERVAL 31 DAY) AND | ||||||
|  |             `type` = 'reminder_sent' | ||||||
|  |         GROUP BY `time_key` | ||||||
|  |         ORDER BY `time_key`" | ||||||
|  |     ) | ||||||
|  |     .fetch_all(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     .unwrap(); | ||||||
|  |  | ||||||
|  |     let history_failed = sqlx::query_as_unchecked!( | ||||||
|  |         TimeFrame, | ||||||
|  |         "SELECT | ||||||
|  |             FROM_UNIXTIME(FLOOR(UNIX_TIMESTAMP(`utc_time`) / 86400) * 86400) AS `time_key`, | ||||||
|  |             COUNT(1) AS `count` | ||||||
|  |         FROM stat | ||||||
|  |         WHERE | ||||||
|  |             `utc_time` > DATE_SUB(NOW(), INTERVAL 31 DAY) AND | ||||||
|  |             `type` = 'reminder_failed' | ||||||
|  |         GROUP BY `time_key` | ||||||
|  |         ORDER BY `time_key`" | ||||||
|  |     ) | ||||||
|  |     .fetch_all(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     .unwrap(); | ||||||
|  |  | ||||||
|  |     let interval_count = sqlx::query!( | ||||||
|  |         "SELECT COUNT(1) AS count | ||||||
|  |         FROM reminders | ||||||
|  |         WHERE | ||||||
|  |             `status` = 'pending' AND ( | ||||||
|  |                 `interval_seconds` IS NOT NULL OR | ||||||
|  |                 `interval_months` IS NOT NULL OR | ||||||
|  |                 `interval_days` IS NOT NULL | ||||||
|  |             )" | ||||||
|  |     ) | ||||||
|  |     .fetch_one(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     .unwrap(); | ||||||
|  |  | ||||||
|  |     let reminder_count = sqlx::query!( | ||||||
|  |         "SELECT COUNT(1) AS count | ||||||
|  |         FROM reminders | ||||||
|  |         WHERE | ||||||
|  |             `status` = 'pending' AND | ||||||
|  |             `interval_seconds` IS NULL AND | ||||||
|  |             `interval_months` IS NULL AND | ||||||
|  |             `interval_days` IS NULL" | ||||||
|  |     ) | ||||||
|  |     .fetch_one(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     .unwrap(); | ||||||
|  |  | ||||||
|  |     Ok(json!({ | ||||||
|  |         "backlog": backlog.backlog, | ||||||
|  |         "scheduleShort": { | ||||||
|  |             "once": schedule_once, | ||||||
|  |             "interval": schedule_interval | ||||||
|  |         }, | ||||||
|  |         "scheduleLong": { | ||||||
|  |             "once": schedule_once_long, | ||||||
|  |             "interval": schedule_interval_long, | ||||||
|  |         }, | ||||||
|  |         "historyLong": { | ||||||
|  |             "sent": history, | ||||||
|  |             "failed": history_failed, | ||||||
|  |         }, | ||||||
|  |         "count": { | ||||||
|  |             "reminders": reminder_count.count, | ||||||
|  |             "intervals": interval_count.count, | ||||||
|  |         } | ||||||
|  |     })) | ||||||
|  | } | ||||||
							
								
								
									
										61
									
								
								web/src/routes/dashboard/api/guild/channels.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,61 @@ | |||||||
|  | use rocket::{http::CookieJar, serde::json::json, State}; | ||||||
|  | use serde::Serialize; | ||||||
|  | use serenity::{ | ||||||
|  |     client::Context, | ||||||
|  |     model::{ | ||||||
|  |         channel::GuildChannel, | ||||||
|  |         id::{ChannelId, GuildId}, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | use crate::{check_authorization, routes::JsonResult}; | ||||||
|  |  | ||||||
|  | #[derive(Serialize)] | ||||||
|  | struct ChannelInfo { | ||||||
|  |     id: String, | ||||||
|  |     name: String, | ||||||
|  |     webhook_avatar: Option<String>, | ||||||
|  |     webhook_name: Option<String>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[get("/api/guild/<id>/channels")] | ||||||
|  | pub async fn get_guild_channels( | ||||||
|  |     id: u64, | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     offline!(Ok(json!(vec![ChannelInfo { | ||||||
|  |         name: "general".to_string(), | ||||||
|  |         id: "1".to_string(), | ||||||
|  |         webhook_avatar: None, | ||||||
|  |         webhook_name: None, | ||||||
|  |     }]))); | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     match GuildId(id).to_guild_cached(ctx.inner()) { | ||||||
|  |         Some(guild) => { | ||||||
|  |             let mut channels = guild | ||||||
|  |                 .channels | ||||||
|  |                 .iter() | ||||||
|  |                 .filter_map(|(id, channel)| channel.to_owned().guild().map(|c| (id.to_owned(), c))) | ||||||
|  |                 .filter(|(_, channel)| channel.is_text_based()) | ||||||
|  |                 .collect::<Vec<(ChannelId, GuildChannel)>>(); | ||||||
|  |  | ||||||
|  |             channels.sort_by(|(_, c1), (_, c2)| c1.position.cmp(&c2.position)); | ||||||
|  |  | ||||||
|  |             let channel_info = channels | ||||||
|  |                 .iter() | ||||||
|  |                 .map(|(channel_id, channel)| ChannelInfo { | ||||||
|  |                     name: channel.name.to_string(), | ||||||
|  |                     id: channel_id.to_string(), | ||||||
|  |                     webhook_avatar: None, | ||||||
|  |                     webhook_name: None, | ||||||
|  |                 }) | ||||||
|  |                 .collect::<Vec<ChannelInfo>>(); | ||||||
|  |  | ||||||
|  |             Ok(json!(channel_info)) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         None => json_err!("Bot not in guild"), | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										42
									
								
								web/src/routes/dashboard/api/guild/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,42 @@ | |||||||
|  | mod channels; | ||||||
|  | mod reminders; | ||||||
|  | mod roles; | ||||||
|  | mod templates; | ||||||
|  |  | ||||||
|  | use std::env; | ||||||
|  |  | ||||||
|  | pub use channels::*; | ||||||
|  | pub use reminders::*; | ||||||
|  | use rocket::{http::CookieJar, serde::json::json, State}; | ||||||
|  | pub use roles::*; | ||||||
|  | use serenity::{ | ||||||
|  |     client::Context, | ||||||
|  |     model::id::{GuildId, RoleId}, | ||||||
|  | }; | ||||||
|  | pub use templates::*; | ||||||
|  |  | ||||||
|  | use crate::{check_authorization, routes::JsonResult}; | ||||||
|  |  | ||||||
|  | #[get("/api/guild/<id>")] | ||||||
|  | pub async fn get_guild_info(id: u64, cookies: &CookieJar<'_>, ctx: &State<Context>) -> JsonResult { | ||||||
|  |     offline!(Ok(json!({ "patreon": true, "name": "Guild" }))); | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     match GuildId(id).to_guild_cached(ctx.inner()) { | ||||||
|  |         Some(guild) => { | ||||||
|  |             let member_res = GuildId(env::var("PATREON_GUILD_ID").unwrap().parse().unwrap()) | ||||||
|  |                 .member(&ctx.inner(), guild.owner_id) | ||||||
|  |                 .await; | ||||||
|  |  | ||||||
|  |             let patreon = member_res.map_or(false, |member| { | ||||||
|  |                 member | ||||||
|  |                     .roles | ||||||
|  |                     .contains(&RoleId(env::var("PATREON_ROLE_ID").unwrap().parse().unwrap())) | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             Ok(json!({ "patreon": patreon, "name": guild.name })) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         None => json_err!("Bot not in guild"), | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										373
									
								
								web/src/routes/dashboard/api/guild/reminders.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,373 @@ | |||||||
|  | use rocket::{ | ||||||
|  |     http::CookieJar, | ||||||
|  |     serde::json::{json, Json}, | ||||||
|  |     State, | ||||||
|  | }; | ||||||
|  | use serenity::{ | ||||||
|  |     client::Context, | ||||||
|  |     model::id::{ChannelId, GuildId, UserId}, | ||||||
|  | }; | ||||||
|  | use sqlx::{MySql, Pool}; | ||||||
|  |  | ||||||
|  | use crate::{ | ||||||
|  |     check_authorization, check_guild_subscription, check_subscription, | ||||||
|  |     consts::MIN_INTERVAL, | ||||||
|  |     guards::transaction::Transaction, | ||||||
|  |     routes::{ | ||||||
|  |         dashboard::{ | ||||||
|  |             create_database_channel, create_reminder, DeleteReminder, PatchReminder, Reminder, | ||||||
|  |         }, | ||||||
|  |         JsonResult, | ||||||
|  |     }, | ||||||
|  |     Database, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #[post("/api/guild/<id>/reminders", data = "<reminder>")] | ||||||
|  | pub async fn create_guild_reminder( | ||||||
|  |     id: u64, | ||||||
|  |     reminder: Json<Reminder>, | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     mut transaction: Transaction<'_>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     let user_id = | ||||||
|  |         cookies.get_private("userid").map(|c| c.value().parse::<u64>().ok()).flatten().unwrap(); | ||||||
|  |  | ||||||
|  |     match create_reminder( | ||||||
|  |         ctx.inner(), | ||||||
|  |         &mut transaction, | ||||||
|  |         GuildId(id), | ||||||
|  |         UserId(user_id), | ||||||
|  |         reminder.into_inner(), | ||||||
|  |     ) | ||||||
|  |     .await | ||||||
|  |     { | ||||||
|  |         Ok(r) => match transaction.commit().await { | ||||||
|  |             Ok(_) => Ok(r), | ||||||
|  |             Err(e) => { | ||||||
|  |                 warn!("Couldn't commit transaction: {:?}", e); | ||||||
|  |                 json_err!("Couldn't commit transaction.") | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |  | ||||||
|  |         Err(e) => Err(e), | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[get("/api/guild/<id>/reminders")] | ||||||
|  | pub async fn get_reminders( | ||||||
|  |     id: u64, | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     pool: &State<Pool<MySql>>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     let channels_res = GuildId(id).channels(&ctx.inner()).await; | ||||||
|  |  | ||||||
|  |     match channels_res { | ||||||
|  |         Ok(channels) => { | ||||||
|  |             let channels = channels | ||||||
|  |                 .keys() | ||||||
|  |                 .into_iter() | ||||||
|  |                 .map(|k| k.as_u64().to_string()) | ||||||
|  |                 .collect::<Vec<String>>() | ||||||
|  |                 .join(","); | ||||||
|  |  | ||||||
|  |             sqlx::query_as_unchecked!( | ||||||
|  |                 Reminder, | ||||||
|  |                 "SELECT | ||||||
|  |                  reminders.attachment, | ||||||
|  |                  reminders.attachment_name, | ||||||
|  |                  reminders.avatar, | ||||||
|  |                  channels.channel, | ||||||
|  |                  reminders.content, | ||||||
|  |                  reminders.embed_author, | ||||||
|  |                  reminders.embed_author_url, | ||||||
|  |                  reminders.embed_color, | ||||||
|  |                  reminders.embed_description, | ||||||
|  |                  reminders.embed_footer, | ||||||
|  |                  reminders.embed_footer_url, | ||||||
|  |                  reminders.embed_image_url, | ||||||
|  |                  reminders.embed_thumbnail_url, | ||||||
|  |                  reminders.embed_title, | ||||||
|  |                  IFNULL(reminders.embed_fields, '[]') AS embed_fields, | ||||||
|  |                  reminders.enabled, | ||||||
|  |                  reminders.expires, | ||||||
|  |                  reminders.interval_seconds, | ||||||
|  |                  reminders.interval_days, | ||||||
|  |                  reminders.interval_months, | ||||||
|  |                  reminders.name, | ||||||
|  |                  reminders.restartable, | ||||||
|  |                  reminders.tts, | ||||||
|  |                  reminders.uid, | ||||||
|  |                  reminders.username, | ||||||
|  |                  reminders.utc_time | ||||||
|  |                 FROM reminders | ||||||
|  |                 LEFT JOIN channels ON channels.id = reminders.channel_id | ||||||
|  |                 WHERE `status` = 'pending' AND FIND_IN_SET(channels.channel, ?)", | ||||||
|  |                 channels | ||||||
|  |             ) | ||||||
|  |             .fetch_all(pool.inner()) | ||||||
|  |             .await | ||||||
|  |             .map(|r| Ok(json!(r))) | ||||||
|  |             .unwrap_or_else(|e| { | ||||||
|  |                 warn!("Failed to complete SQL query: {:?}", e); | ||||||
|  |  | ||||||
|  |                 json_err!("Could not load reminders") | ||||||
|  |             }) | ||||||
|  |         } | ||||||
|  |         Err(e) => { | ||||||
|  |             warn!("Could not fetch channels from {}: {:?}", id, e); | ||||||
|  |  | ||||||
|  |             Ok(json!([])) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[patch("/api/guild/<id>/reminders", data = "<reminder>")] | ||||||
|  | pub async fn edit_reminder( | ||||||
|  |     id: u64, | ||||||
|  |     reminder: Json<PatchReminder>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     mut transaction: Transaction<'_>, | ||||||
|  |     pool: &State<Pool<Database>>, | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     let mut error = vec![]; | ||||||
|  |  | ||||||
|  |     let user_id = | ||||||
|  |         cookies.get_private("userid").map(|c| c.value().parse::<u64>().ok()).flatten().unwrap(); | ||||||
|  |  | ||||||
|  |     if reminder.message_ok() { | ||||||
|  |         update_field!(transaction.executor(), error, reminder.[ | ||||||
|  |             content, | ||||||
|  |             embed_author, | ||||||
|  |             embed_description, | ||||||
|  |             embed_footer, | ||||||
|  |             embed_title, | ||||||
|  |             embed_fields, | ||||||
|  |             username | ||||||
|  |         ]); | ||||||
|  |     } else { | ||||||
|  |         error.push("Message exceeds limits.".to_string()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     update_field!(transaction.executor(), error, reminder.[ | ||||||
|  |         attachment, | ||||||
|  |         attachment_name, | ||||||
|  |         avatar, | ||||||
|  |         embed_author_url, | ||||||
|  |         embed_color, | ||||||
|  |         embed_footer_url, | ||||||
|  |         embed_image_url, | ||||||
|  |         embed_thumbnail_url, | ||||||
|  |         enabled, | ||||||
|  |         expires, | ||||||
|  |         name, | ||||||
|  |         restartable, | ||||||
|  |         tts, | ||||||
|  |         utc_time | ||||||
|  |     ]); | ||||||
|  |  | ||||||
|  |     if reminder.interval_days.flatten().is_some() | ||||||
|  |         || reminder.interval_months.flatten().is_some() | ||||||
|  |         || reminder.interval_seconds.flatten().is_some() | ||||||
|  |     { | ||||||
|  |         if check_guild_subscription(&ctx.inner(), id).await | ||||||
|  |             || check_subscription(&ctx.inner(), user_id).await | ||||||
|  |         { | ||||||
|  |             let new_interval_length = match reminder.interval_days { | ||||||
|  |                 Some(interval) => interval.unwrap_or(0), | ||||||
|  |                 None => sqlx::query!( | ||||||
|  |                     "SELECT interval_days AS days FROM reminders WHERE uid = ?", | ||||||
|  |                     reminder.uid | ||||||
|  |                 ) | ||||||
|  |                 .fetch_one(transaction.executor()) | ||||||
|  |                 .await | ||||||
|  |                 .map_err(|e| { | ||||||
|  |                     warn!("Error updating reminder interval: {:?}", e); | ||||||
|  |                     json!({ "reminder": Option::<Reminder>::None, "errors": vec!["Unknown error"] }) | ||||||
|  |                 })? | ||||||
|  |                 .days | ||||||
|  |                 .unwrap_or(0), | ||||||
|  |             } * 86400 + match reminder.interval_months { | ||||||
|  |                 Some(interval) => interval.unwrap_or(0), | ||||||
|  |                 None => sqlx::query!( | ||||||
|  |                     "SELECT interval_months AS months FROM reminders WHERE uid = ?", | ||||||
|  |                     reminder.uid | ||||||
|  |                 ) | ||||||
|  |                 .fetch_one(transaction.executor()) | ||||||
|  |                 .await | ||||||
|  |                 .map_err(|e| { | ||||||
|  |                     warn!("Error updating reminder interval: {:?}", e); | ||||||
|  |                     json!({ "reminder": Option::<Reminder>::None, "errors": vec!["Unknown error"] }) | ||||||
|  |                 })? | ||||||
|  |                 .months | ||||||
|  |                 .unwrap_or(0), | ||||||
|  |             } * 2592000 + match reminder.interval_seconds { | ||||||
|  |                 Some(interval) => interval.unwrap_or(0), | ||||||
|  |                 None => sqlx::query!( | ||||||
|  |                     "SELECT interval_seconds AS seconds FROM reminders WHERE uid = ?", | ||||||
|  |                     reminder.uid | ||||||
|  |                 ) | ||||||
|  |                 .fetch_one(transaction.executor()) | ||||||
|  |                 .await | ||||||
|  |                 .map_err(|e| { | ||||||
|  |                     warn!("Error updating reminder interval: {:?}", e); | ||||||
|  |                     json!({ "reminder": Option::<Reminder>::None, "errors": vec!["Unknown error"] }) | ||||||
|  |                 })? | ||||||
|  |                 .seconds | ||||||
|  |                 .unwrap_or(0), | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             if new_interval_length < *MIN_INTERVAL { | ||||||
|  |                 error.push(String::from("New interval is too short.")); | ||||||
|  |             } else { | ||||||
|  |                 update_field!(transaction.executor(), error, reminder.[ | ||||||
|  |                     interval_days, | ||||||
|  |                     interval_months, | ||||||
|  |                     interval_seconds | ||||||
|  |                 ]); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if reminder.channel > 0 { | ||||||
|  |         let channel = ChannelId(reminder.channel).to_channel_cached(&ctx.inner()); | ||||||
|  |         match channel { | ||||||
|  |             Some(channel) => { | ||||||
|  |                 let channel_matches_guild = channel.guild().map_or(false, |c| c.guild_id.0 == id); | ||||||
|  |  | ||||||
|  |                 if !channel_matches_guild { | ||||||
|  |                     warn!( | ||||||
|  |                         "Error in `edit_reminder`: channel {:?} not found for guild {}", | ||||||
|  |                         reminder.channel, id | ||||||
|  |                     ); | ||||||
|  |  | ||||||
|  |                     return Err(json!({"error": "Channel not found"})); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 let channel = create_database_channel( | ||||||
|  |                     ctx.inner(), | ||||||
|  |                     ChannelId(reminder.channel), | ||||||
|  |                     &mut transaction, | ||||||
|  |                 ) | ||||||
|  |                 .await; | ||||||
|  |  | ||||||
|  |                 if let Err(e) = channel { | ||||||
|  |                     warn!("`create_database_channel` returned an error code: {:?}", e); | ||||||
|  |  | ||||||
|  |                     return Err( | ||||||
|  |                         json!({"error": "Failed to configure channel for reminders. Please check the bot permissions"}), | ||||||
|  |                     ); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 let channel = channel.unwrap(); | ||||||
|  |  | ||||||
|  |                 match sqlx::query!( | ||||||
|  |                     "UPDATE reminders SET channel_id = ? WHERE uid = ?", | ||||||
|  |                     channel, | ||||||
|  |                     reminder.uid | ||||||
|  |                 ) | ||||||
|  |                 .execute(transaction.executor()) | ||||||
|  |                 .await | ||||||
|  |                 { | ||||||
|  |                     Ok(_) => {} | ||||||
|  |                     Err(e) => { | ||||||
|  |                         warn!("Error setting channel: {:?}", e); | ||||||
|  |  | ||||||
|  |                         error.push("Couldn't set channel".to_string()) | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             None => { | ||||||
|  |                 warn!( | ||||||
|  |                     "Error in `edit_reminder`: channel {:?} not found for guild {}", | ||||||
|  |                     reminder.channel, id | ||||||
|  |                 ); | ||||||
|  |  | ||||||
|  |                 return Err(json!({"error": "Channel not found"})); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if let Err(e) = transaction.commit().await { | ||||||
|  |         warn!("Couldn't commit transaction: {:?}", e); | ||||||
|  |         return json_err!("Couldn't commit transaction"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     match sqlx::query_as_unchecked!( | ||||||
|  |         Reminder, | ||||||
|  |         "SELECT reminders.attachment, | ||||||
|  |          reminders.attachment_name, | ||||||
|  |          reminders.avatar, | ||||||
|  |          channels.channel, | ||||||
|  |          reminders.content, | ||||||
|  |          reminders.embed_author, | ||||||
|  |          reminders.embed_author_url, | ||||||
|  |          reminders.embed_color, | ||||||
|  |          reminders.embed_description, | ||||||
|  |          reminders.embed_footer, | ||||||
|  |          reminders.embed_footer_url, | ||||||
|  |          reminders.embed_image_url, | ||||||
|  |          reminders.embed_thumbnail_url, | ||||||
|  |          reminders.embed_title, | ||||||
|  |          reminders.embed_fields, | ||||||
|  |          reminders.enabled, | ||||||
|  |          reminders.expires, | ||||||
|  |          reminders.interval_seconds, | ||||||
|  |          reminders.interval_days, | ||||||
|  |          reminders.interval_months, | ||||||
|  |          reminders.name, | ||||||
|  |          reminders.restartable, | ||||||
|  |          reminders.tts, | ||||||
|  |          reminders.uid, | ||||||
|  |          reminders.username, | ||||||
|  |          reminders.utc_time | ||||||
|  |         FROM reminders | ||||||
|  |         LEFT JOIN channels ON channels.id = reminders.channel_id | ||||||
|  |         WHERE uid = ?", | ||||||
|  |         reminder.uid | ||||||
|  |     ) | ||||||
|  |     .fetch_one(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     { | ||||||
|  |         Ok(reminder) => Ok(json!({"reminder": reminder, "errors": error})), | ||||||
|  |  | ||||||
|  |         Err(e) => { | ||||||
|  |             warn!("Error exiting `edit_reminder': {:?}", e); | ||||||
|  |  | ||||||
|  |             Err(json!({"reminder": Option::<Reminder>::None, "errors": vec!["Unknown error"]})) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[delete("/api/guild/<id>/reminders", data = "<reminder>")] | ||||||
|  | pub async fn delete_reminder( | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     id: u64, | ||||||
|  |     reminder: Json<DeleteReminder>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     pool: &State<Pool<MySql>>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     match sqlx::query!("UPDATE reminders SET `status` = 'deleted' WHERE uid = ?", reminder.uid) | ||||||
|  |         .execute(pool.inner()) | ||||||
|  |         .await | ||||||
|  |     { | ||||||
|  |         Ok(_) => Ok(json!({})), | ||||||
|  |  | ||||||
|  |         Err(e) => { | ||||||
|  |             warn!("Error in `delete_reminder`: {:?}", e); | ||||||
|  |  | ||||||
|  |             Err(json!({"error": "Could not delete reminder"})) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										35
									
								
								web/src/routes/dashboard/api/guild/roles.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,35 @@ | |||||||
|  | use rocket::{http::CookieJar, serde::json::json, State}; | ||||||
|  | use serde::Serialize; | ||||||
|  | use serenity::client::Context; | ||||||
|  |  | ||||||
|  | use crate::{check_authorization, routes::JsonResult}; | ||||||
|  |  | ||||||
|  | #[derive(Serialize)] | ||||||
|  | struct RoleInfo { | ||||||
|  |     id: String, | ||||||
|  |     name: String, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[get("/api/guild/<id>/roles")] | ||||||
|  | pub async fn get_guild_roles(id: u64, cookies: &CookieJar<'_>, ctx: &State<Context>) -> JsonResult { | ||||||
|  |     offline!(Ok(json!(vec![RoleInfo { name: "@everyone".to_string(), id: "1".to_string() }]))); | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     let roles_res = ctx.cache.guild_roles(id); | ||||||
|  |  | ||||||
|  |     match roles_res { | ||||||
|  |         Some(roles) => { | ||||||
|  |             let roles = roles | ||||||
|  |                 .iter() | ||||||
|  |                 .map(|(_, r)| RoleInfo { id: r.id.to_string(), name: r.name.to_string() }) | ||||||
|  |                 .collect::<Vec<RoleInfo>>(); | ||||||
|  |  | ||||||
|  |             Ok(json!(roles)) | ||||||
|  |         } | ||||||
|  |         None => { | ||||||
|  |             warn!("Could not fetch roles from {}", id); | ||||||
|  |  | ||||||
|  |             json_err!("Could not get roles") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										181
									
								
								web/src/routes/dashboard/api/guild/templates.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,181 @@ | |||||||
|  | use rocket::{ | ||||||
|  |     http::CookieJar, | ||||||
|  |     serde::json::{json, Json}, | ||||||
|  |     State, | ||||||
|  | }; | ||||||
|  | use serenity::client::Context; | ||||||
|  | use sqlx::{MySql, Pool}; | ||||||
|  |  | ||||||
|  | use crate::{ | ||||||
|  |     check_authorization, | ||||||
|  |     consts::{ | ||||||
|  |         MAX_CONTENT_LENGTH, MAX_EMBED_AUTHOR_LENGTH, MAX_EMBED_DESCRIPTION_LENGTH, | ||||||
|  |         MAX_EMBED_FIELDS, MAX_EMBED_FIELD_TITLE_LENGTH, MAX_EMBED_FIELD_VALUE_LENGTH, | ||||||
|  |         MAX_EMBED_FOOTER_LENGTH, MAX_EMBED_TITLE_LENGTH, MAX_URL_LENGTH, MAX_USERNAME_LENGTH, | ||||||
|  |     }, | ||||||
|  |     routes::{ | ||||||
|  |         dashboard::{template_name_default, DeleteReminderTemplate, ReminderTemplate}, | ||||||
|  |         JsonResult, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #[get("/api/guild/<id>/templates")] | ||||||
|  | pub async fn get_reminder_templates( | ||||||
|  |     id: u64, | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     pool: &State<Pool<MySql>>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     match sqlx::query_as_unchecked!( | ||||||
|  |         ReminderTemplate, | ||||||
|  |         "SELECT * FROM reminder_template WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?)", | ||||||
|  |         id | ||||||
|  |     ) | ||||||
|  |     .fetch_all(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     { | ||||||
|  |         Ok(templates) => Ok(json!(templates)), | ||||||
|  |         Err(e) => { | ||||||
|  |             warn!("Could not fetch templates from {}: {:?}", id, e); | ||||||
|  |  | ||||||
|  |             json_err!("Could not get templates") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[post("/api/guild/<id>/templates", data = "<reminder_template>")] | ||||||
|  | pub async fn create_reminder_template( | ||||||
|  |     id: u64, | ||||||
|  |     reminder_template: Json<ReminderTemplate>, | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     pool: &State<Pool<MySql>>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     // validate lengths | ||||||
|  |     check_length!(MAX_CONTENT_LENGTH, reminder_template.content); | ||||||
|  |     check_length!(MAX_EMBED_DESCRIPTION_LENGTH, reminder_template.embed_description); | ||||||
|  |     check_length!(MAX_EMBED_TITLE_LENGTH, reminder_template.embed_title); | ||||||
|  |     check_length!(MAX_EMBED_AUTHOR_LENGTH, reminder_template.embed_author); | ||||||
|  |     check_length!(MAX_EMBED_FOOTER_LENGTH, reminder_template.embed_footer); | ||||||
|  |     check_length_opt!(MAX_EMBED_FIELDS, reminder_template.embed_fields); | ||||||
|  |     if let Some(fields) = &reminder_template.embed_fields { | ||||||
|  |         for field in &fields.0 { | ||||||
|  |             check_length!(MAX_EMBED_FIELD_VALUE_LENGTH, field.value); | ||||||
|  |             check_length!(MAX_EMBED_FIELD_TITLE_LENGTH, field.title); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     check_length_opt!(MAX_USERNAME_LENGTH, reminder_template.username); | ||||||
|  |     check_length_opt!( | ||||||
|  |         MAX_URL_LENGTH, | ||||||
|  |         reminder_template.embed_footer_url, | ||||||
|  |         reminder_template.embed_thumbnail_url, | ||||||
|  |         reminder_template.embed_author_url, | ||||||
|  |         reminder_template.embed_image_url, | ||||||
|  |         reminder_template.avatar | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     // validate urls | ||||||
|  |     check_url_opt!( | ||||||
|  |         reminder_template.embed_footer_url, | ||||||
|  |         reminder_template.embed_thumbnail_url, | ||||||
|  |         reminder_template.embed_author_url, | ||||||
|  |         reminder_template.embed_image_url, | ||||||
|  |         reminder_template.avatar | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     let name = if reminder_template.name.is_empty() { | ||||||
|  |         template_name_default() | ||||||
|  |     } else { | ||||||
|  |         reminder_template.name.clone() | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     match sqlx::query!( | ||||||
|  |         "INSERT INTO reminder_template | ||||||
|  |         (guild_id, | ||||||
|  |          name, | ||||||
|  |          attachment, | ||||||
|  |          attachment_name, | ||||||
|  |          avatar, | ||||||
|  |          content, | ||||||
|  |          embed_author, | ||||||
|  |          embed_author_url, | ||||||
|  |          embed_color, | ||||||
|  |          embed_description, | ||||||
|  |          embed_footer, | ||||||
|  |          embed_footer_url, | ||||||
|  |          embed_image_url, | ||||||
|  |          embed_thumbnail_url, | ||||||
|  |          embed_title, | ||||||
|  |          embed_fields, | ||||||
|  |          interval_seconds, | ||||||
|  |          interval_days, | ||||||
|  |          interval_months, | ||||||
|  |          tts, | ||||||
|  |          username | ||||||
|  |         ) VALUES ((SELECT id FROM guilds WHERE guild = ?), ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, | ||||||
|  |          ?, ?, ?, ?, ?, ?, ?)", | ||||||
|  |         id, | ||||||
|  |         name, | ||||||
|  |         reminder_template.attachment, | ||||||
|  |         reminder_template.attachment_name, | ||||||
|  |         reminder_template.avatar, | ||||||
|  |         reminder_template.content, | ||||||
|  |         reminder_template.embed_author, | ||||||
|  |         reminder_template.embed_author_url, | ||||||
|  |         reminder_template.embed_color, | ||||||
|  |         reminder_template.embed_description, | ||||||
|  |         reminder_template.embed_footer, | ||||||
|  |         reminder_template.embed_footer_url, | ||||||
|  |         reminder_template.embed_image_url, | ||||||
|  |         reminder_template.embed_thumbnail_url, | ||||||
|  |         reminder_template.embed_title, | ||||||
|  |         reminder_template.embed_fields, | ||||||
|  |         reminder_template.interval_seconds, | ||||||
|  |         reminder_template.interval_days, | ||||||
|  |         reminder_template.interval_months, | ||||||
|  |         reminder_template.tts, | ||||||
|  |         reminder_template.username, | ||||||
|  |     ) | ||||||
|  |     .fetch_all(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     { | ||||||
|  |         Ok(_) => Ok(json!({})), | ||||||
|  |         Err(e) => { | ||||||
|  |             warn!("Could not create template for {}: {:?}", id, e); | ||||||
|  |  | ||||||
|  |             json_err!("Could not create template") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[delete("/api/guild/<id>/templates", data = "<delete_reminder_template>")] | ||||||
|  | pub async fn delete_reminder_template( | ||||||
|  |     id: u64, | ||||||
|  |     delete_reminder_template: Json<DeleteReminderTemplate>, | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     pool: &State<Pool<MySql>>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     match sqlx::query!( | ||||||
|  |         "DELETE FROM reminder_template WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?) AND id = ?", | ||||||
|  |         id, delete_reminder_template.id | ||||||
|  |     ) | ||||||
|  |     .fetch_all(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     { | ||||||
|  |         Ok(_) => { | ||||||
|  |             Ok(json!({})) | ||||||
|  |         } | ||||||
|  |         Err(e) => { | ||||||
|  |             warn!("Could not delete template from {}: {:?}", id, e); | ||||||
|  |  | ||||||
|  |             json_err!("Could not delete template") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								web/src/routes/dashboard/api/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,2 @@ | |||||||
|  | pub mod guild; | ||||||
|  | pub mod user; | ||||||
| @@ -1,36 +1,14 @@ | |||||||
| use std::env; |  | ||||||
| 
 |  | ||||||
| use chrono_tz::Tz; |  | ||||||
| use reqwest::Client; | use reqwest::Client; | ||||||
| use rocket::{ | use rocket::{ | ||||||
|     http::CookieJar, |     http::CookieJar, | ||||||
|     serde::json::{json, Json, Value as JsonValue}, |     serde::json::{json, Value as JsonValue}, | ||||||
|     State, |     State, | ||||||
| }; | }; | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
| use serenity::{ | use serenity::model::{id::GuildId, permissions::Permissions}; | ||||||
|     client::Context, |  | ||||||
|     model::{ |  | ||||||
|         id::{GuildId, RoleId}, |  | ||||||
|         permissions::Permissions, |  | ||||||
|     }, |  | ||||||
| }; |  | ||||||
| use sqlx::{MySql, Pool}; |  | ||||||
| 
 | 
 | ||||||
| use crate::consts::DISCORD_API; | use crate::consts::DISCORD_API; | ||||||
| 
 | 
 | ||||||
| #[derive(Serialize)] |  | ||||||
| struct UserInfo { |  | ||||||
|     name: String, |  | ||||||
|     patreon: bool, |  | ||||||
|     timezone: Option<String>, |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #[derive(Deserialize)] |  | ||||||
| pub struct UpdateUser { |  | ||||||
|     timezone: String, |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #[derive(Serialize)] | #[derive(Serialize)] | ||||||
| struct GuildInfo { | struct GuildInfo { | ||||||
|     id: String, |     id: String, | ||||||
| @@ -38,9 +16,8 @@ struct GuildInfo { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(Deserialize)] | #[derive(Deserialize)] | ||||||
| pub struct PartialGuild { | struct PartialGuild { | ||||||
|     pub id: GuildId, |     pub id: GuildId, | ||||||
|     pub icon: Option<String>, |  | ||||||
|     pub name: String, |     pub name: String, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|     pub owner: bool, |     pub owner: bool, | ||||||
| @@ -48,71 +25,10 @@ pub struct PartialGuild { | |||||||
|     pub permissions: Option<String>, |     pub permissions: Option<String>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[get("/api/user")] |  | ||||||
| pub async fn get_user_info( |  | ||||||
|     cookies: &CookieJar<'_>, |  | ||||||
|     ctx: &State<Context>, |  | ||||||
|     pool: &State<Pool<MySql>>, |  | ||||||
| ) -> JsonValue { |  | ||||||
|     if let Some(user_id) = |  | ||||||
|         cookies.get_private("userid").map(|u| u.value().parse::<u64>().ok()).flatten() |  | ||||||
|     { |  | ||||||
|         let member_res = GuildId(env::var("PATREON_GUILD_ID").unwrap().parse().unwrap()) |  | ||||||
|             .member(&ctx.inner(), user_id) |  | ||||||
|             .await; |  | ||||||
| 
 |  | ||||||
|         let timezone = sqlx::query!("SELECT timezone FROM users WHERE user = ?", user_id) |  | ||||||
|             .fetch_one(pool.inner()) |  | ||||||
|             .await |  | ||||||
|             .map_or(None, |q| Some(q.timezone)); |  | ||||||
| 
 |  | ||||||
|         let user_info = UserInfo { |  | ||||||
|             name: cookies |  | ||||||
|                 .get_private("username") |  | ||||||
|                 .map_or("DiscordUser#0000".to_string(), |c| c.value().to_string()), |  | ||||||
|             patreon: member_res.map_or(false, |member| { |  | ||||||
|                 member |  | ||||||
|                     .roles |  | ||||||
|                     .contains(&RoleId(env::var("PATREON_ROLE_ID").unwrap().parse().unwrap())) |  | ||||||
|             }), |  | ||||||
|             timezone, |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         json!(user_info) |  | ||||||
|     } else { |  | ||||||
|         json!({"error": "Not authorized"}) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #[patch("/api/user", data = "<user>")] |  | ||||||
| pub async fn update_user_info( |  | ||||||
|     cookies: &CookieJar<'_>, |  | ||||||
|     user: Json<UpdateUser>, |  | ||||||
|     pool: &State<Pool<MySql>>, |  | ||||||
| ) -> JsonValue { |  | ||||||
|     if let Some(user_id) = |  | ||||||
|         cookies.get_private("userid").map(|u| u.value().parse::<u64>().ok()).flatten() |  | ||||||
|     { |  | ||||||
|         if user.timezone.parse::<Tz>().is_ok() { |  | ||||||
|             let _ = sqlx::query!( |  | ||||||
|                 "UPDATE users SET timezone = ? WHERE user = ?", |  | ||||||
|                 user.timezone, |  | ||||||
|                 user_id, |  | ||||||
|             ) |  | ||||||
|             .execute(pool.inner()) |  | ||||||
|             .await; |  | ||||||
| 
 |  | ||||||
|             json!({}) |  | ||||||
|         } else { |  | ||||||
|             json!({"error": "Timezone not recognized"}) |  | ||||||
|         } |  | ||||||
|     } else { |  | ||||||
|         json!({"error": "Not authorized"}) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #[get("/api/user/guilds")] | #[get("/api/user/guilds")] | ||||||
| pub async fn get_user_guilds(cookies: &CookieJar<'_>, reqwest_client: &State<Client>) -> JsonValue { | pub async fn get_user_guilds(cookies: &CookieJar<'_>, reqwest_client: &State<Client>) -> JsonValue { | ||||||
|  |     offline!(json!(vec![GuildInfo { id: "1".to_string(), name: "Guild".to_string() }])); | ||||||
|  | 
 | ||||||
|     if let Some(access_token) = cookies.get_private("access_token") { |     if let Some(access_token) = cookies.get_private("access_token") { | ||||||
|         let request_res = reqwest_client |         let request_res = reqwest_client | ||||||
|             .get(format!("{}/users/@me/guilds", DISCORD_API)) |             .get(format!("{}/users/@me/guilds", DISCORD_API)) | ||||||
							
								
								
									
										97
									
								
								web/src/routes/dashboard/api/user/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,97 @@ | |||||||
|  | mod guilds; | ||||||
|  |  | ||||||
|  | use std::env; | ||||||
|  |  | ||||||
|  | use chrono_tz::Tz; | ||||||
|  | pub use guilds::*; | ||||||
|  | use rocket::{ | ||||||
|  |     http::CookieJar, | ||||||
|  |     serde::json::{json, Json, Value as JsonValue}, | ||||||
|  |     State, | ||||||
|  | }; | ||||||
|  | use serde::{Deserialize, Serialize}; | ||||||
|  | use serenity::{ | ||||||
|  |     client::Context, | ||||||
|  |     model::id::{GuildId, RoleId}, | ||||||
|  | }; | ||||||
|  | use sqlx::{MySql, Pool}; | ||||||
|  |  | ||||||
|  | #[derive(Serialize)] | ||||||
|  | struct UserInfo { | ||||||
|  |     name: String, | ||||||
|  |     patreon: bool, | ||||||
|  |     timezone: Option<String>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Deserialize)] | ||||||
|  | pub struct UpdateUser { | ||||||
|  |     timezone: String, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[get("/api/user")] | ||||||
|  | pub async fn get_user_info( | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     pool: &State<Pool<MySql>>, | ||||||
|  | ) -> JsonValue { | ||||||
|  |     offline!(json!(UserInfo { name: "Discord".to_string(), patreon: true, timezone: None })); | ||||||
|  |  | ||||||
|  |     if let Some(user_id) = | ||||||
|  |         cookies.get_private("userid").map(|u| u.value().parse::<u64>().ok()).flatten() | ||||||
|  |     { | ||||||
|  |         let member_res = GuildId(env::var("PATREON_GUILD_ID").unwrap().parse().unwrap()) | ||||||
|  |             .member(&ctx.inner(), user_id) | ||||||
|  |             .await; | ||||||
|  |  | ||||||
|  |         let timezone = sqlx::query!( | ||||||
|  |             "SELECT IFNULL(timezone, 'UTC') AS timezone FROM users WHERE user = ?", | ||||||
|  |             user_id | ||||||
|  |         ) | ||||||
|  |         .fetch_one(pool.inner()) | ||||||
|  |         .await | ||||||
|  |         .map_or(None, |q| Some(q.timezone)); | ||||||
|  |  | ||||||
|  |         let user_info = UserInfo { | ||||||
|  |             name: cookies | ||||||
|  |                 .get_private("username") | ||||||
|  |                 .map_or("DiscordUser#0000".to_string(), |c| c.value().to_string()), | ||||||
|  |             patreon: member_res.map_or(false, |member| { | ||||||
|  |                 member | ||||||
|  |                     .roles | ||||||
|  |                     .contains(&RoleId(env::var("PATREON_ROLE_ID").unwrap().parse().unwrap())) | ||||||
|  |             }), | ||||||
|  |             timezone, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         json!(user_info) | ||||||
|  |     } else { | ||||||
|  |         json!({"error": "Not authorized"}) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[patch("/api/user", data = "<user>")] | ||||||
|  | pub async fn update_user_info( | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     user: Json<UpdateUser>, | ||||||
|  |     pool: &State<Pool<MySql>>, | ||||||
|  | ) -> JsonValue { | ||||||
|  |     if let Some(user_id) = | ||||||
|  |         cookies.get_private("userid").map(|u| u.value().parse::<u64>().ok()).flatten() | ||||||
|  |     { | ||||||
|  |         if user.timezone.parse::<Tz>().is_ok() { | ||||||
|  |             let _ = sqlx::query!( | ||||||
|  |                 "UPDATE users SET timezone = ? WHERE user = ?", | ||||||
|  |                 user.timezone, | ||||||
|  |                 user_id, | ||||||
|  |             ) | ||||||
|  |             .execute(pool.inner()) | ||||||
|  |             .await; | ||||||
|  |  | ||||||
|  |             json!({}) | ||||||
|  |         } else { | ||||||
|  |             json!({"error": "Timezone not recognized"}) | ||||||
|  |         } | ||||||
|  |     } else { | ||||||
|  |         json!({"error": "Not authorized"}) | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										20
									
								
								web/src/routes/dashboard/api/user/models.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | |||||||
|  | use std::env; | ||||||
|  |  | ||||||
|  | use chrono_tz::Tz; | ||||||
|  | use reqwest::Client; | ||||||
|  | use rocket::{ | ||||||
|  |     http::CookieJar, | ||||||
|  |     serde::json::{json, Json, Value as JsonValue}, | ||||||
|  |     State, | ||||||
|  | }; | ||||||
|  | use serde::{Deserialize, Serialize}; | ||||||
|  | use serenity::{ | ||||||
|  |     client::Context, | ||||||
|  |     model::{ | ||||||
|  |         id::{GuildId, RoleId}, | ||||||
|  |         permissions::Permissions, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  | use sqlx::{MySql, Pool}; | ||||||
|  |  | ||||||
|  | use crate::{consts::DISCORD_API, routes::JsonResult}; | ||||||
							
								
								
									
										29
									
								
								web/src/routes/dashboard/api/user/reminders.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,29 @@ | |||||||
|  | use std::env; | ||||||
|  |  | ||||||
|  | use chrono_tz::Tz; | ||||||
|  | use reqwest::Client; | ||||||
|  | use rocket::{ | ||||||
|  |     http::CookieJar, | ||||||
|  |     serde::json::{json, Json, Value as JsonValue}, | ||||||
|  |     State, | ||||||
|  | }; | ||||||
|  | use serde::{Deserialize, Serialize}; | ||||||
|  | use serenity::{ | ||||||
|  |     client::Context, | ||||||
|  |     model::{ | ||||||
|  |         id::{GuildId, RoleId}, | ||||||
|  |         permissions::Permissions, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  | use sqlx::{MySql, Pool}; | ||||||
|  |  | ||||||
|  | use crate::{consts::DISCORD_API, routes::JsonResult}; | ||||||
|  |  | ||||||
|  | #[get("/api/user/reminders")] | ||||||
|  | pub async fn get_reminders( | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     pool: &State<Pool<MySql>>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     Ok(json! {}) | ||||||
|  | } | ||||||
							
								
								
									
										447
									
								
								web/src/routes/dashboard/export.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,447 @@ | |||||||
|  | use csv::{QuoteStyle, WriterBuilder}; | ||||||
|  | use rocket::{ | ||||||
|  |     http::CookieJar, | ||||||
|  |     serde::json::{json, serde_json, Json}, | ||||||
|  |     State, | ||||||
|  | }; | ||||||
|  | use serenity::{ | ||||||
|  |     client::Context, | ||||||
|  |     model::id::{ChannelId, GuildId, UserId}, | ||||||
|  | }; | ||||||
|  | use sqlx::{MySql, Pool}; | ||||||
|  |  | ||||||
|  | use crate::{ | ||||||
|  |     check_authorization, | ||||||
|  |     guards::transaction::Transaction, | ||||||
|  |     routes::{ | ||||||
|  |         dashboard::{ | ||||||
|  |             create_reminder, generate_uid, ImportBody, Reminder, ReminderCsv, ReminderTemplateCsv, | ||||||
|  |             TodoCsv, | ||||||
|  |         }, | ||||||
|  |         JsonResult, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #[get("/api/guild/<id>/export/reminders")] | ||||||
|  | pub async fn export_reminders( | ||||||
|  |     id: u64, | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     pool: &State<Pool<MySql>>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     let mut csv_writer = WriterBuilder::new().quote_style(QuoteStyle::Always).from_writer(vec![]); | ||||||
|  |  | ||||||
|  |     let channels_res = GuildId(id).channels(&ctx.inner()).await; | ||||||
|  |  | ||||||
|  |     match channels_res { | ||||||
|  |         Ok(channels) => { | ||||||
|  |             let channels = channels | ||||||
|  |                 .keys() | ||||||
|  |                 .into_iter() | ||||||
|  |                 .map(|k| k.as_u64().to_string()) | ||||||
|  |                 .collect::<Vec<String>>() | ||||||
|  |                 .join(","); | ||||||
|  |  | ||||||
|  |             let result = sqlx::query_as_unchecked!( | ||||||
|  |                 ReminderCsv, | ||||||
|  |                 "SELECT | ||||||
|  |                  reminders.attachment, | ||||||
|  |                  reminders.attachment_name, | ||||||
|  |                  reminders.avatar, | ||||||
|  |                  CONCAT('#', channels.channel) AS channel, | ||||||
|  |                  reminders.content, | ||||||
|  |                  reminders.embed_author, | ||||||
|  |                  reminders.embed_author_url, | ||||||
|  |                  reminders.embed_color, | ||||||
|  |                  reminders.embed_description, | ||||||
|  |                  reminders.embed_footer, | ||||||
|  |                  reminders.embed_footer_url, | ||||||
|  |                  reminders.embed_image_url, | ||||||
|  |                  reminders.embed_thumbnail_url, | ||||||
|  |                  reminders.embed_title, | ||||||
|  |                  reminders.embed_fields, | ||||||
|  |                  reminders.enabled, | ||||||
|  |                  reminders.expires, | ||||||
|  |                  reminders.interval_seconds, | ||||||
|  |                  reminders.interval_days, | ||||||
|  |                  reminders.interval_months, | ||||||
|  |                  reminders.name, | ||||||
|  |                  reminders.restartable, | ||||||
|  |                  reminders.tts, | ||||||
|  |                  reminders.username, | ||||||
|  |                  reminders.utc_time | ||||||
|  |                 FROM reminders | ||||||
|  |                 LEFT JOIN channels ON channels.id = reminders.channel_id | ||||||
|  |                 WHERE FIND_IN_SET(channels.channel, ?) AND status = 'pending'", | ||||||
|  |                 channels | ||||||
|  |             ) | ||||||
|  |             .fetch_all(pool.inner()) | ||||||
|  |             .await; | ||||||
|  |  | ||||||
|  |             match result { | ||||||
|  |                 Ok(reminders) => { | ||||||
|  |                     reminders.iter().for_each(|reminder| { | ||||||
|  |                         csv_writer.serialize(reminder).unwrap(); | ||||||
|  |                     }); | ||||||
|  |  | ||||||
|  |                     match csv_writer.into_inner() { | ||||||
|  |                         Ok(inner) => match String::from_utf8(inner) { | ||||||
|  |                             Ok(encoded) => Ok(json!({ "body": encoded })), | ||||||
|  |  | ||||||
|  |                             Err(e) => { | ||||||
|  |                                 warn!("Failed to write UTF-8: {:?}", e); | ||||||
|  |  | ||||||
|  |                                 Err(json!({"error": "Failed to write UTF-8"})) | ||||||
|  |                             } | ||||||
|  |                         }, | ||||||
|  |  | ||||||
|  |                         Err(e) => { | ||||||
|  |                             warn!("Failed to extract CSV: {:?}", e); | ||||||
|  |  | ||||||
|  |                             Err(json!({"error": "Failed to extract CSV"})) | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 Err(e) => { | ||||||
|  |                     warn!("Failed to complete SQL query: {:?}", e); | ||||||
|  |  | ||||||
|  |                     Err(json!({"error": "Failed to query reminders"})) | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         Err(e) => { | ||||||
|  |             warn!("Could not fetch channels from {}: {:?}", id, e); | ||||||
|  |  | ||||||
|  |             Err(json!({"error": "Failed to get guild channels"})) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[put("/api/guild/<id>/export/reminders", data = "<body>")] | ||||||
|  | pub(crate) async fn import_reminders( | ||||||
|  |     id: u64, | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     body: Json<ImportBody>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     mut transaction: Transaction<'_>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     let user_id = | ||||||
|  |         cookies.get_private("userid").map(|c| c.value().parse::<u64>().ok()).flatten().unwrap(); | ||||||
|  |  | ||||||
|  |     match base64::decode(&body.body) { | ||||||
|  |         Ok(body) => { | ||||||
|  |             let mut reader = csv::Reader::from_reader(body.as_slice()); | ||||||
|  |             let mut count = 0; | ||||||
|  |  | ||||||
|  |             for result in reader.deserialize::<ReminderCsv>() { | ||||||
|  |                 match result { | ||||||
|  |                     Ok(record) => { | ||||||
|  |                         let channel_id = record.channel.split_at(1).1; | ||||||
|  |  | ||||||
|  |                         match channel_id.parse::<u64>() { | ||||||
|  |                             Ok(channel_id) => { | ||||||
|  |                                 let reminder = Reminder { | ||||||
|  |                                     attachment: record.attachment, | ||||||
|  |                                     attachment_name: record.attachment_name, | ||||||
|  |                                     avatar: record.avatar, | ||||||
|  |                                     channel: channel_id, | ||||||
|  |                                     content: record.content, | ||||||
|  |                                     embed_author: record.embed_author, | ||||||
|  |                                     embed_author_url: record.embed_author_url, | ||||||
|  |                                     embed_color: record.embed_color, | ||||||
|  |                                     embed_description: record.embed_description, | ||||||
|  |                                     embed_footer: record.embed_footer, | ||||||
|  |                                     embed_footer_url: record.embed_footer_url, | ||||||
|  |                                     embed_image_url: record.embed_image_url, | ||||||
|  |                                     embed_thumbnail_url: record.embed_thumbnail_url, | ||||||
|  |                                     embed_title: record.embed_title, | ||||||
|  |                                     embed_fields: record | ||||||
|  |                                         .embed_fields | ||||||
|  |                                         .map(|s| serde_json::from_str(&s).ok()) | ||||||
|  |                                         .flatten(), | ||||||
|  |                                     enabled: record.enabled, | ||||||
|  |                                     expires: record.expires, | ||||||
|  |                                     interval_seconds: record.interval_seconds, | ||||||
|  |                                     interval_days: record.interval_days, | ||||||
|  |                                     interval_months: record.interval_months, | ||||||
|  |                                     name: record.name, | ||||||
|  |                                     restartable: record.restartable, | ||||||
|  |                                     tts: record.tts, | ||||||
|  |                                     uid: generate_uid(), | ||||||
|  |                                     username: record.username, | ||||||
|  |                                     utc_time: record.utc_time, | ||||||
|  |                                 }; | ||||||
|  |  | ||||||
|  |                                 create_reminder( | ||||||
|  |                                     ctx.inner(), | ||||||
|  |                                     &mut transaction, | ||||||
|  |                                     GuildId(id), | ||||||
|  |                                     UserId(user_id), | ||||||
|  |                                     reminder, | ||||||
|  |                                 ) | ||||||
|  |                                 .await?; | ||||||
|  |  | ||||||
|  |                                 count += 1; | ||||||
|  |                             } | ||||||
|  |  | ||||||
|  |                             Err(_) => { | ||||||
|  |                                 return json_err!(format!( | ||||||
|  |                                     "Failed to parse channel {}", | ||||||
|  |                                     channel_id | ||||||
|  |                                 )); | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     Err(e) => { | ||||||
|  |                         warn!("Couldn't deserialize CSV row: {:?}", e); | ||||||
|  |  | ||||||
|  |                         return json_err!("Deserialize error. Aborted"); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             match transaction.commit().await { | ||||||
|  |                 Ok(_) => Ok(json!({ | ||||||
|  |                     "message": format!("Imported {} reminders", count) | ||||||
|  |                 })), | ||||||
|  |  | ||||||
|  |                 Err(e) => { | ||||||
|  |                     warn!("Failed to commit transaction: {:?}", e); | ||||||
|  |                     json_err!("Couldn't commit transaction") | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         Err(_) => { | ||||||
|  |             json_err!("Malformed base64") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[get("/api/guild/<id>/export/todos")] | ||||||
|  | pub async fn export_todos( | ||||||
|  |     id: u64, | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     pool: &State<Pool<MySql>>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     let mut csv_writer = WriterBuilder::new().quote_style(QuoteStyle::Always).from_writer(vec![]); | ||||||
|  |  | ||||||
|  |     match sqlx::query_as_unchecked!( | ||||||
|  |         TodoCsv, | ||||||
|  |         "SELECT value, CONCAT('#', channels.channel) AS channel_id FROM todos | ||||||
|  |         LEFT JOIN channels ON todos.channel_id = channels.id | ||||||
|  |         INNER JOIN guilds ON todos.guild_id = guilds.id | ||||||
|  |         WHERE guilds.guild = ?", | ||||||
|  |         id | ||||||
|  |     ) | ||||||
|  |     .fetch_all(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     { | ||||||
|  |         Ok(todos) => { | ||||||
|  |             todos.iter().for_each(|todo| { | ||||||
|  |                 csv_writer.serialize(todo).unwrap(); | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             match csv_writer.into_inner() { | ||||||
|  |                 Ok(inner) => match String::from_utf8(inner) { | ||||||
|  |                     Ok(encoded) => Ok(json!({ "body": encoded })), | ||||||
|  |  | ||||||
|  |                     Err(e) => { | ||||||
|  |                         warn!("Failed to write UTF-8: {:?}", e); | ||||||
|  |  | ||||||
|  |                         json_err!("Failed to write UTF-8") | ||||||
|  |                     } | ||||||
|  |                 }, | ||||||
|  |  | ||||||
|  |                 Err(e) => { | ||||||
|  |                     warn!("Failed to extract CSV: {:?}", e); | ||||||
|  |  | ||||||
|  |                     json_err!("Failed to extract CSV") | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         Err(e) => { | ||||||
|  |             warn!("Could not fetch templates from {}: {:?}", id, e); | ||||||
|  |  | ||||||
|  |             json_err!("Failed to query templates") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[put("/api/guild/<id>/export/todos", data = "<body>")] | ||||||
|  | pub async fn import_todos( | ||||||
|  |     id: u64, | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     body: Json<ImportBody>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     pool: &State<Pool<MySql>>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     let channels_res = GuildId(id).channels(&ctx.inner()).await; | ||||||
|  |  | ||||||
|  |     match channels_res { | ||||||
|  |         Ok(channels) => match base64::decode(&body.body) { | ||||||
|  |             Ok(body) => { | ||||||
|  |                 let mut reader = csv::Reader::from_reader(body.as_slice()); | ||||||
|  |  | ||||||
|  |                 let query_placeholder = "(?, (SELECT id FROM channels WHERE channel = ?), (SELECT id FROM guilds WHERE guild = ?))"; | ||||||
|  |                 let mut query_params = vec![]; | ||||||
|  |  | ||||||
|  |                 for result in reader.deserialize::<TodoCsv>() { | ||||||
|  |                     match result { | ||||||
|  |                         Ok(record) => match record.channel_id { | ||||||
|  |                             Some(channel_id) => { | ||||||
|  |                                 let channel_id = channel_id.split_at(1).1; | ||||||
|  |  | ||||||
|  |                                 match channel_id.parse::<u64>() { | ||||||
|  |                                     Ok(channel_id) => { | ||||||
|  |                                         if channels.contains_key(&ChannelId(channel_id)) { | ||||||
|  |                                             query_params.push((record.value, Some(channel_id), id)); | ||||||
|  |                                         } else { | ||||||
|  |                                             return json_err!(format!( | ||||||
|  |                                                 "Invalid channel ID {}", | ||||||
|  |                                                 channel_id | ||||||
|  |                                             )); | ||||||
|  |                                         } | ||||||
|  |                                     } | ||||||
|  |  | ||||||
|  |                                     Err(_) => { | ||||||
|  |                                         return json_err!(format!( | ||||||
|  |                                             "Invalid channel ID {}", | ||||||
|  |                                             channel_id | ||||||
|  |                                         )); | ||||||
|  |                                     } | ||||||
|  |                                 } | ||||||
|  |                             } | ||||||
|  |  | ||||||
|  |                             None => { | ||||||
|  |                                 query_params.push((record.value, None, id)); | ||||||
|  |                             } | ||||||
|  |                         }, | ||||||
|  |  | ||||||
|  |                         Err(e) => { | ||||||
|  |                             warn!("Couldn't deserialize CSV row: {:?}", e); | ||||||
|  |  | ||||||
|  |                             return json_err!("Deserialize error. Aborted"); | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 let query_str = format!( | ||||||
|  |                     "INSERT INTO todos (value, channel_id, guild_id) VALUES {}", | ||||||
|  |                     vec![query_placeholder].repeat(query_params.len()).join(",") | ||||||
|  |                 ); | ||||||
|  |                 let mut query = sqlx::query(&query_str); | ||||||
|  |  | ||||||
|  |                 for param in query_params { | ||||||
|  |                     query = query.bind(param.0).bind(param.1).bind(param.2); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 let res = query.execute(pool.inner()).await; | ||||||
|  |  | ||||||
|  |                 match res { | ||||||
|  |                     Ok(_) => Ok(json!({})), | ||||||
|  |  | ||||||
|  |                     Err(e) => { | ||||||
|  |                         warn!("Couldn't execute todo query: {:?}", e); | ||||||
|  |  | ||||||
|  |                         json_err!("An unexpected error occured.") | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             Err(_) => { | ||||||
|  |                 json_err!("Malformed base64") | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |  | ||||||
|  |         Err(e) => { | ||||||
|  |             warn!("Couldn't fetch channels for guild {}: {:?}", id, e); | ||||||
|  |  | ||||||
|  |             json_err!("Couldn't fetch channels.") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[get("/api/guild/<id>/export/reminder_templates")] | ||||||
|  | pub async fn export_reminder_templates( | ||||||
|  |     id: u64, | ||||||
|  |     cookies: &CookieJar<'_>, | ||||||
|  |     ctx: &State<Context>, | ||||||
|  |     pool: &State<Pool<MySql>>, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|  |     let mut csv_writer = WriterBuilder::new().quote_style(QuoteStyle::Always).from_writer(vec![]); | ||||||
|  |  | ||||||
|  |     match sqlx::query_as_unchecked!( | ||||||
|  |         ReminderTemplateCsv, | ||||||
|  |         "SELECT | ||||||
|  |          name, | ||||||
|  |          attachment, | ||||||
|  |          attachment_name, | ||||||
|  |          avatar, | ||||||
|  |          content, | ||||||
|  |          embed_author, | ||||||
|  |          embed_author_url, | ||||||
|  |          embed_color, | ||||||
|  |          embed_description, | ||||||
|  |          embed_footer, | ||||||
|  |          embed_footer_url, | ||||||
|  |          embed_image_url, | ||||||
|  |          embed_thumbnail_url, | ||||||
|  |          embed_title, | ||||||
|  |          embed_fields, | ||||||
|  |          interval_seconds, | ||||||
|  |          interval_days, | ||||||
|  |          interval_months, | ||||||
|  |          tts, | ||||||
|  |          username | ||||||
|  |         FROM reminder_template WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?)", | ||||||
|  |         id | ||||||
|  |     ) | ||||||
|  |     .fetch_all(pool.inner()) | ||||||
|  |     .await | ||||||
|  |     { | ||||||
|  |         Ok(templates) => { | ||||||
|  |             templates.iter().for_each(|template| { | ||||||
|  |                 csv_writer.serialize(template).unwrap(); | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             match csv_writer.into_inner() { | ||||||
|  |                 Ok(inner) => match String::from_utf8(inner) { | ||||||
|  |                     Ok(encoded) => Ok(json!({ "body": encoded })), | ||||||
|  |  | ||||||
|  |                     Err(e) => { | ||||||
|  |                         warn!("Failed to write UTF-8: {:?}", e); | ||||||
|  |  | ||||||
|  |                         json_err!("Failed to write UTF-8") | ||||||
|  |                     } | ||||||
|  |                 }, | ||||||
|  |  | ||||||
|  |                 Err(e) => { | ||||||
|  |                     warn!("Failed to extract CSV: {:?}", e); | ||||||
|  |  | ||||||
|  |                     json_err!("Failed to extract CSV") | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         Err(e) => { | ||||||
|  |             warn!("Could not fetch templates from {}: {:?}", id, e); | ||||||
|  |  | ||||||
|  |             json_err!("Failed to query templates") | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,733 +0,0 @@ | |||||||
| use std::env; |  | ||||||
|  |  | ||||||
| use base64; |  | ||||||
| use chrono::Utc; |  | ||||||
| use rocket::{ |  | ||||||
|     http::CookieJar, |  | ||||||
|     serde::json::{json, Json, Value as JsonValue}, |  | ||||||
|     State, |  | ||||||
| }; |  | ||||||
| use serde::Serialize; |  | ||||||
| use serenity::{ |  | ||||||
|     client::Context, |  | ||||||
|     model::{ |  | ||||||
|         channel::GuildChannel, |  | ||||||
|         id::{ChannelId, GuildId, RoleId}, |  | ||||||
|     }, |  | ||||||
| }; |  | ||||||
| use sqlx::{MySql, Pool}; |  | ||||||
|  |  | ||||||
| use crate::{ |  | ||||||
|     check_guild_subscription, check_subscription, |  | ||||||
|     consts::{ |  | ||||||
|         DAY, MAX_CONTENT_LENGTH, MAX_EMBED_AUTHOR_LENGTH, MAX_EMBED_DESCRIPTION_LENGTH, |  | ||||||
|         MAX_EMBED_FIELDS, MAX_EMBED_FIELD_TITLE_LENGTH, MAX_EMBED_FIELD_VALUE_LENGTH, |  | ||||||
|         MAX_EMBED_FOOTER_LENGTH, MAX_EMBED_TITLE_LENGTH, MAX_URL_LENGTH, MAX_USERNAME_LENGTH, |  | ||||||
|         MIN_INTERVAL, |  | ||||||
|     }, |  | ||||||
|     routes::dashboard::{ |  | ||||||
|         create_database_channel, generate_uid, name_default, template_name_default, DeleteReminder, |  | ||||||
|         DeleteReminderTemplate, PatchReminder, Reminder, ReminderTemplate, |  | ||||||
|     }, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| #[derive(Serialize)] |  | ||||||
| struct ChannelInfo { |  | ||||||
|     id: String, |  | ||||||
|     name: String, |  | ||||||
|     webhook_avatar: Option<String>, |  | ||||||
|     webhook_name: Option<String>, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[get("/api/guild/<id>/patreon")] |  | ||||||
| pub async fn get_guild_patreon( |  | ||||||
|     id: u64, |  | ||||||
|     cookies: &CookieJar<'_>, |  | ||||||
|     ctx: &State<Context>, |  | ||||||
| ) -> JsonValue { |  | ||||||
|     check_authorization!(cookies, ctx.inner(), id); |  | ||||||
|  |  | ||||||
|     match GuildId(id).to_guild_cached(ctx.inner()) { |  | ||||||
|         Some(guild) => { |  | ||||||
|             let member_res = GuildId(env::var("PATREON_GUILD_ID").unwrap().parse().unwrap()) |  | ||||||
|                 .member(&ctx.inner(), guild.owner_id) |  | ||||||
|                 .await; |  | ||||||
|  |  | ||||||
|             let patreon = member_res.map_or(false, |member| { |  | ||||||
|                 member |  | ||||||
|                     .roles |  | ||||||
|                     .contains(&RoleId(env::var("PATREON_ROLE_ID").unwrap().parse().unwrap())) |  | ||||||
|             }); |  | ||||||
|  |  | ||||||
|             json!({ "patreon": patreon }) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         None => { |  | ||||||
|             json!({"error": "Bot not in guild"}) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[get("/api/guild/<id>/channels")] |  | ||||||
| pub async fn get_guild_channels( |  | ||||||
|     id: u64, |  | ||||||
|     cookies: &CookieJar<'_>, |  | ||||||
|     ctx: &State<Context>, |  | ||||||
| ) -> JsonValue { |  | ||||||
|     check_authorization!(cookies, ctx.inner(), id); |  | ||||||
|  |  | ||||||
|     match GuildId(id).to_guild_cached(ctx.inner()) { |  | ||||||
|         Some(guild) => { |  | ||||||
|             let mut channels = guild |  | ||||||
|                 .channels |  | ||||||
|                 .iter() |  | ||||||
|                 .filter_map(|(id, channel)| channel.to_owned().guild().map(|c| (id.to_owned(), c))) |  | ||||||
|                 .filter(|(_, channel)| channel.is_text_based()) |  | ||||||
|                 .collect::<Vec<(ChannelId, GuildChannel)>>(); |  | ||||||
|  |  | ||||||
|             channels.sort_by(|(_, c1), (_, c2)| c1.position.cmp(&c2.position)); |  | ||||||
|  |  | ||||||
|             let channel_info = channels |  | ||||||
|                 .iter() |  | ||||||
|                 .map(|(channel_id, channel)| ChannelInfo { |  | ||||||
|                     name: channel.name.to_string(), |  | ||||||
|                     id: channel_id.to_string(), |  | ||||||
|                     webhook_avatar: None, |  | ||||||
|                     webhook_name: None, |  | ||||||
|                 }) |  | ||||||
|                 .collect::<Vec<ChannelInfo>>(); |  | ||||||
|  |  | ||||||
|             json!(channel_info) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         None => { |  | ||||||
|             json!({"error": "Bot not in guild"}) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[derive(Serialize)] |  | ||||||
| struct RoleInfo { |  | ||||||
|     id: String, |  | ||||||
|     name: String, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[get("/api/guild/<id>/roles")] |  | ||||||
| pub async fn get_guild_roles(id: u64, cookies: &CookieJar<'_>, ctx: &State<Context>) -> JsonValue { |  | ||||||
|     check_authorization!(cookies, ctx.inner(), id); |  | ||||||
|  |  | ||||||
|     let roles_res = ctx.cache.guild_roles(id); |  | ||||||
|  |  | ||||||
|     match roles_res { |  | ||||||
|         Some(roles) => { |  | ||||||
|             let roles = roles |  | ||||||
|                 .iter() |  | ||||||
|                 .map(|(_, r)| RoleInfo { id: r.id.to_string(), name: r.name.to_string() }) |  | ||||||
|                 .collect::<Vec<RoleInfo>>(); |  | ||||||
|  |  | ||||||
|             json!(roles) |  | ||||||
|         } |  | ||||||
|         None => { |  | ||||||
|             warn!("Could not fetch roles from {}", id); |  | ||||||
|  |  | ||||||
|             json!({"error": "Could not get roles"}) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[get("/api/guild/<id>/templates")] |  | ||||||
| pub async fn get_reminder_templates( |  | ||||||
|     id: u64, |  | ||||||
|     cookies: &CookieJar<'_>, |  | ||||||
|     ctx: &State<Context>, |  | ||||||
|     pool: &State<Pool<MySql>>, |  | ||||||
| ) -> JsonValue { |  | ||||||
|     check_authorization!(cookies, ctx.inner(), id); |  | ||||||
|  |  | ||||||
|     match sqlx::query_as_unchecked!( |  | ||||||
|         ReminderTemplate, |  | ||||||
|         "SELECT * FROM reminder_template WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?)", |  | ||||||
|         id |  | ||||||
|     ) |  | ||||||
|     .fetch_all(pool.inner()) |  | ||||||
|     .await |  | ||||||
|     { |  | ||||||
|         Ok(templates) => { |  | ||||||
|             json!(templates) |  | ||||||
|         } |  | ||||||
|         Err(e) => { |  | ||||||
|             warn!("Could not fetch templates from {}: {:?}", id, e); |  | ||||||
|  |  | ||||||
|             json!({"error": "Could not get templates"}) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[post("/api/guild/<id>/templates", data = "<reminder_template>")] |  | ||||||
| pub async fn create_reminder_template( |  | ||||||
|     id: u64, |  | ||||||
|     reminder_template: Json<ReminderTemplate>, |  | ||||||
|     cookies: &CookieJar<'_>, |  | ||||||
|     ctx: &State<Context>, |  | ||||||
|     pool: &State<Pool<MySql>>, |  | ||||||
| ) -> JsonValue { |  | ||||||
|     check_authorization!(cookies, ctx.inner(), id); |  | ||||||
|  |  | ||||||
|     // validate lengths |  | ||||||
|     check_length!(MAX_CONTENT_LENGTH, reminder_template.content); |  | ||||||
|     check_length!(MAX_EMBED_DESCRIPTION_LENGTH, reminder_template.embed_description); |  | ||||||
|     check_length!(MAX_EMBED_TITLE_LENGTH, reminder_template.embed_title); |  | ||||||
|     check_length!(MAX_EMBED_AUTHOR_LENGTH, reminder_template.embed_author); |  | ||||||
|     check_length!(MAX_EMBED_FOOTER_LENGTH, reminder_template.embed_footer); |  | ||||||
|     check_length_opt!(MAX_EMBED_FIELDS, reminder_template.embed_fields); |  | ||||||
|     if let Some(fields) = &reminder_template.embed_fields { |  | ||||||
|         for field in &fields.0 { |  | ||||||
|             check_length!(MAX_EMBED_FIELD_VALUE_LENGTH, field.value); |  | ||||||
|             check_length!(MAX_EMBED_FIELD_TITLE_LENGTH, field.title); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     check_length_opt!(MAX_USERNAME_LENGTH, reminder_template.username); |  | ||||||
|     check_length_opt!( |  | ||||||
|         MAX_URL_LENGTH, |  | ||||||
|         reminder_template.embed_footer_url, |  | ||||||
|         reminder_template.embed_thumbnail_url, |  | ||||||
|         reminder_template.embed_author_url, |  | ||||||
|         reminder_template.embed_image_url, |  | ||||||
|         reminder_template.avatar |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     // validate urls |  | ||||||
|     check_url_opt!( |  | ||||||
|         reminder_template.embed_footer_url, |  | ||||||
|         reminder_template.embed_thumbnail_url, |  | ||||||
|         reminder_template.embed_author_url, |  | ||||||
|         reminder_template.embed_image_url, |  | ||||||
|         reminder_template.avatar |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     let name = if reminder_template.name.is_empty() { |  | ||||||
|         template_name_default() |  | ||||||
|     } else { |  | ||||||
|         reminder_template.name.clone() |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     match sqlx::query!( |  | ||||||
|         "INSERT INTO reminder_template |  | ||||||
|         (guild_id, |  | ||||||
|          name, |  | ||||||
|          attachment, |  | ||||||
|          attachment_name, |  | ||||||
|          avatar, |  | ||||||
|          content, |  | ||||||
|          embed_author, |  | ||||||
|          embed_author_url, |  | ||||||
|          embed_color, |  | ||||||
|          embed_description, |  | ||||||
|          embed_footer, |  | ||||||
|          embed_footer_url, |  | ||||||
|          embed_image_url, |  | ||||||
|          embed_thumbnail_url, |  | ||||||
|          embed_title, |  | ||||||
|          embed_fields, |  | ||||||
|          tts, |  | ||||||
|          username |  | ||||||
|         ) VALUES ((SELECT id FROM guilds WHERE guild = ?), ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)", |  | ||||||
|         id, name, |  | ||||||
|         reminder_template.attachment, |  | ||||||
|         reminder_template.attachment_name, |  | ||||||
|         reminder_template.avatar, |  | ||||||
|         reminder_template.content, |  | ||||||
|         reminder_template.embed_author, |  | ||||||
|         reminder_template.embed_author_url, |  | ||||||
|         reminder_template.embed_color, |  | ||||||
|         reminder_template.embed_description, |  | ||||||
|         reminder_template.embed_footer, |  | ||||||
|         reminder_template.embed_footer_url, |  | ||||||
|         reminder_template.embed_image_url, |  | ||||||
|         reminder_template.embed_thumbnail_url, |  | ||||||
|         reminder_template.embed_title, |  | ||||||
|         reminder_template.embed_fields, |  | ||||||
|         reminder_template.tts, |  | ||||||
|         reminder_template.username, |  | ||||||
|     ) |  | ||||||
|     .fetch_all(pool.inner()) |  | ||||||
|     .await |  | ||||||
|     { |  | ||||||
|         Ok(_) => { |  | ||||||
|             json!({}) |  | ||||||
|         } |  | ||||||
|         Err(e) => { |  | ||||||
|             warn!("Could not fetch templates from {}: {:?}", id, e); |  | ||||||
|  |  | ||||||
|             json!({"error": "Could not get templates"}) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[delete("/api/guild/<id>/templates", data = "<delete_reminder_template>")] |  | ||||||
| pub async fn delete_reminder_template( |  | ||||||
|     id: u64, |  | ||||||
|     delete_reminder_template: Json<DeleteReminderTemplate>, |  | ||||||
|     cookies: &CookieJar<'_>, |  | ||||||
|     ctx: &State<Context>, |  | ||||||
|     pool: &State<Pool<MySql>>, |  | ||||||
| ) -> JsonValue { |  | ||||||
|     check_authorization!(cookies, ctx.inner(), id); |  | ||||||
|  |  | ||||||
|     match sqlx::query!( |  | ||||||
|         "DELETE FROM reminder_template WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?) AND id = ?", |  | ||||||
|         id, delete_reminder_template.id |  | ||||||
|     ) |  | ||||||
|     .fetch_all(pool.inner()) |  | ||||||
|     .await |  | ||||||
|     { |  | ||||||
|         Ok(_) => { |  | ||||||
|             json!({}) |  | ||||||
|         } |  | ||||||
|         Err(e) => { |  | ||||||
|             warn!("Could not delete template from {}: {:?}", id, e); |  | ||||||
|  |  | ||||||
|             json!({"error": "Could not delete template"}) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[post("/api/guild/<id>/reminders", data = "<reminder>")] |  | ||||||
| pub async fn create_reminder( |  | ||||||
|     id: u64, |  | ||||||
|     reminder: Json<Reminder>, |  | ||||||
|     cookies: &CookieJar<'_>, |  | ||||||
|     serenity_context: &State<Context>, |  | ||||||
|     pool: &State<Pool<MySql>>, |  | ||||||
| ) -> JsonValue { |  | ||||||
|     check_authorization!(cookies, serenity_context.inner(), id); |  | ||||||
|  |  | ||||||
|     let user_id = |  | ||||||
|         cookies.get_private("userid").map(|c| c.value().parse::<u64>().ok()).flatten().unwrap(); |  | ||||||
|  |  | ||||||
|     // validate channel |  | ||||||
|     let channel = ChannelId(reminder.channel).to_channel_cached(&serenity_context.inner()); |  | ||||||
|     let channel_exists = channel.is_some(); |  | ||||||
|  |  | ||||||
|     let channel_matches_guild = |  | ||||||
|         channel.map_or(false, |c| c.guild().map_or(false, |c| c.guild_id.0 == id)); |  | ||||||
|  |  | ||||||
|     if !channel_matches_guild || !channel_exists { |  | ||||||
|         warn!( |  | ||||||
|             "Error in `create_reminder`: channel {} not found for guild {} (channel exists: {})", |  | ||||||
|             reminder.channel, id, channel_exists |  | ||||||
|         ); |  | ||||||
|  |  | ||||||
|         return json!({"error": "Channel not found"}); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     let channel = create_database_channel( |  | ||||||
|         serenity_context.inner(), |  | ||||||
|         ChannelId(reminder.channel), |  | ||||||
|         pool.inner(), |  | ||||||
|     ) |  | ||||||
|     .await; |  | ||||||
|  |  | ||||||
|     if let Err(e) = channel { |  | ||||||
|         warn!("`create_database_channel` returned an error code: {:?}", e); |  | ||||||
|  |  | ||||||
|         return json!({"error": "Failed to configure channel for reminders. Please check the bot permissions"}); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     let channel = channel.unwrap(); |  | ||||||
|  |  | ||||||
|     // validate lengths |  | ||||||
|     check_length!(MAX_CONTENT_LENGTH, reminder.content); |  | ||||||
|     check_length!(MAX_EMBED_DESCRIPTION_LENGTH, reminder.embed_description); |  | ||||||
|     check_length!(MAX_EMBED_TITLE_LENGTH, reminder.embed_title); |  | ||||||
|     check_length!(MAX_EMBED_AUTHOR_LENGTH, reminder.embed_author); |  | ||||||
|     check_length!(MAX_EMBED_FOOTER_LENGTH, reminder.embed_footer); |  | ||||||
|     check_length_opt!(MAX_EMBED_FIELDS, reminder.embed_fields); |  | ||||||
|     if let Some(fields) = &reminder.embed_fields { |  | ||||||
|         for field in &fields.0 { |  | ||||||
|             check_length!(MAX_EMBED_FIELD_VALUE_LENGTH, field.value); |  | ||||||
|             check_length!(MAX_EMBED_FIELD_TITLE_LENGTH, field.title); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     check_length_opt!(MAX_USERNAME_LENGTH, reminder.username); |  | ||||||
|     check_length_opt!( |  | ||||||
|         MAX_URL_LENGTH, |  | ||||||
|         reminder.embed_footer_url, |  | ||||||
|         reminder.embed_thumbnail_url, |  | ||||||
|         reminder.embed_author_url, |  | ||||||
|         reminder.embed_image_url, |  | ||||||
|         reminder.avatar |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     // validate urls |  | ||||||
|     check_url_opt!( |  | ||||||
|         reminder.embed_footer_url, |  | ||||||
|         reminder.embed_thumbnail_url, |  | ||||||
|         reminder.embed_author_url, |  | ||||||
|         reminder.embed_image_url, |  | ||||||
|         reminder.avatar |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     // validate time and interval |  | ||||||
|     if reminder.utc_time < Utc::now().naive_utc() { |  | ||||||
|         return json!({"error": "Time must be in the future"}); |  | ||||||
|     } |  | ||||||
|     if reminder.interval_seconds.is_some() || reminder.interval_months.is_some() { |  | ||||||
|         if reminder.interval_months.unwrap_or(0) * 30 * DAY as u32 |  | ||||||
|             + reminder.interval_seconds.unwrap_or(0) |  | ||||||
|             < *MIN_INTERVAL |  | ||||||
|         { |  | ||||||
|             return json!({"error": "Interval too short"}); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // check patreon if necessary |  | ||||||
|     if reminder.interval_seconds.is_some() || reminder.interval_months.is_some() { |  | ||||||
|         if !check_guild_subscription(serenity_context.inner(), GuildId(id)).await |  | ||||||
|             && !check_subscription(serenity_context.inner(), user_id).await |  | ||||||
|         { |  | ||||||
|             return json!({"error": "Patreon is required to set intervals"}); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // base64 decode error dropped here |  | ||||||
|     let attachment_data = reminder.attachment.as_ref().map(|s| base64::decode(s).ok()).flatten(); |  | ||||||
|     let name = if reminder.name.is_empty() { name_default() } else { reminder.name.clone() }; |  | ||||||
|  |  | ||||||
|     let new_uid = generate_uid(); |  | ||||||
|  |  | ||||||
|     // write to db |  | ||||||
|     match sqlx::query!( |  | ||||||
|         "INSERT INTO reminders ( |  | ||||||
|          uid, |  | ||||||
|          attachment, |  | ||||||
|          attachment_name, |  | ||||||
|          channel_id, |  | ||||||
|          avatar, |  | ||||||
|          content, |  | ||||||
|          embed_author, |  | ||||||
|          embed_author_url, |  | ||||||
|          embed_color, |  | ||||||
|          embed_description, |  | ||||||
|          embed_footer, |  | ||||||
|          embed_footer_url, |  | ||||||
|          embed_image_url, |  | ||||||
|          embed_thumbnail_url, |  | ||||||
|          embed_title, |  | ||||||
|          embed_fields, |  | ||||||
|          enabled, |  | ||||||
|          expires, |  | ||||||
|          interval_seconds, |  | ||||||
|          interval_months, |  | ||||||
|          name, |  | ||||||
|          pin, |  | ||||||
|          restartable, |  | ||||||
|          tts, |  | ||||||
|          username, |  | ||||||
|          `utc_time` |  | ||||||
|         ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)", |  | ||||||
|         new_uid, |  | ||||||
|         attachment_data, |  | ||||||
|         reminder.attachment_name, |  | ||||||
|         channel, |  | ||||||
|         reminder.avatar, |  | ||||||
|         reminder.content, |  | ||||||
|         reminder.embed_author, |  | ||||||
|         reminder.embed_author_url, |  | ||||||
|         reminder.embed_color, |  | ||||||
|         reminder.embed_description, |  | ||||||
|         reminder.embed_footer, |  | ||||||
|         reminder.embed_footer_url, |  | ||||||
|         reminder.embed_image_url, |  | ||||||
|         reminder.embed_thumbnail_url, |  | ||||||
|         reminder.embed_title, |  | ||||||
|         reminder.embed_fields, |  | ||||||
|         reminder.enabled, |  | ||||||
|         reminder.expires, |  | ||||||
|         reminder.interval_seconds, |  | ||||||
|         reminder.interval_months, |  | ||||||
|         name, |  | ||||||
|         reminder.pin, |  | ||||||
|         reminder.restartable, |  | ||||||
|         reminder.tts, |  | ||||||
|         reminder.username, |  | ||||||
|         reminder.utc_time, |  | ||||||
|     ) |  | ||||||
|     .execute(pool.inner()) |  | ||||||
|     .await |  | ||||||
|     { |  | ||||||
|         Ok(_) => sqlx::query_as_unchecked!( |  | ||||||
|             Reminder, |  | ||||||
|             "SELECT |  | ||||||
|              reminders.attachment, |  | ||||||
|              reminders.attachment_name, |  | ||||||
|              reminders.avatar, |  | ||||||
|              channels.channel, |  | ||||||
|              reminders.content, |  | ||||||
|              reminders.embed_author, |  | ||||||
|              reminders.embed_author_url, |  | ||||||
|              reminders.embed_color, |  | ||||||
|              reminders.embed_description, |  | ||||||
|              reminders.embed_footer, |  | ||||||
|              reminders.embed_footer_url, |  | ||||||
|              reminders.embed_image_url, |  | ||||||
|              reminders.embed_thumbnail_url, |  | ||||||
|              reminders.embed_title, |  | ||||||
|              reminders.embed_fields, |  | ||||||
|              reminders.enabled, |  | ||||||
|              reminders.expires, |  | ||||||
|              reminders.interval_seconds, |  | ||||||
|              reminders.interval_months, |  | ||||||
|              reminders.name, |  | ||||||
|              reminders.pin, |  | ||||||
|              reminders.restartable, |  | ||||||
|              reminders.tts, |  | ||||||
|              reminders.uid, |  | ||||||
|              reminders.username, |  | ||||||
|              reminders.utc_time |  | ||||||
|             FROM reminders |  | ||||||
|             LEFT JOIN channels ON channels.id = reminders.channel_id |  | ||||||
|             WHERE uid = ?", |  | ||||||
|             new_uid |  | ||||||
|         ) |  | ||||||
|         .fetch_one(pool.inner()) |  | ||||||
|         .await |  | ||||||
|         .map(|r| json!(r)) |  | ||||||
|         .unwrap_or_else(|e| { |  | ||||||
|             warn!("Failed to complete SQL query: {:?}", e); |  | ||||||
|  |  | ||||||
|             json!({"error": "Could not load reminder"}) |  | ||||||
|         }), |  | ||||||
|  |  | ||||||
|         Err(e) => { |  | ||||||
|             warn!("Error in `create_reminder`: Could not execute query: {:?}", e); |  | ||||||
|  |  | ||||||
|             json!({"error": "Unknown error"}) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[get("/api/guild/<id>/reminders")] |  | ||||||
| pub async fn get_reminders(id: u64, ctx: &State<Context>, pool: &State<Pool<MySql>>) -> JsonValue { |  | ||||||
|     let channels_res = GuildId(id).channels(&ctx.inner()).await; |  | ||||||
|  |  | ||||||
|     match channels_res { |  | ||||||
|         Ok(channels) => { |  | ||||||
|             let channels = channels |  | ||||||
|                 .keys() |  | ||||||
|                 .into_iter() |  | ||||||
|                 .map(|k| k.as_u64().to_string()) |  | ||||||
|                 .collect::<Vec<String>>() |  | ||||||
|                 .join(","); |  | ||||||
|  |  | ||||||
|             sqlx::query_as_unchecked!( |  | ||||||
|                 Reminder, |  | ||||||
|                 "SELECT |  | ||||||
|                  reminders.attachment, |  | ||||||
|                  reminders.attachment_name, |  | ||||||
|                  reminders.avatar, |  | ||||||
|                  channels.channel, |  | ||||||
|                  reminders.content, |  | ||||||
|                  reminders.embed_author, |  | ||||||
|                  reminders.embed_author_url, |  | ||||||
|                  reminders.embed_color, |  | ||||||
|                  reminders.embed_description, |  | ||||||
|                  reminders.embed_footer, |  | ||||||
|                  reminders.embed_footer_url, |  | ||||||
|                  reminders.embed_image_url, |  | ||||||
|                  reminders.embed_thumbnail_url, |  | ||||||
|                  reminders.embed_title, |  | ||||||
|                  reminders.embed_fields, |  | ||||||
|                  reminders.enabled, |  | ||||||
|                  reminders.expires, |  | ||||||
|                  reminders.interval_seconds, |  | ||||||
|                  reminders.interval_months, |  | ||||||
|                  reminders.name, |  | ||||||
|                  reminders.pin, |  | ||||||
|                  reminders.restartable, |  | ||||||
|                  reminders.tts, |  | ||||||
|                  reminders.uid, |  | ||||||
|                  reminders.username, |  | ||||||
|                  reminders.utc_time |  | ||||||
|                 FROM reminders |  | ||||||
|                 LEFT JOIN channels ON channels.id = reminders.channel_id |  | ||||||
|                 WHERE FIND_IN_SET(channels.channel, ?)", |  | ||||||
|                 channels |  | ||||||
|             ) |  | ||||||
|             .fetch_all(pool.inner()) |  | ||||||
|             .await |  | ||||||
|             .map(|r| json!(r)) |  | ||||||
|             .unwrap_or_else(|e| { |  | ||||||
|                 warn!("Failed to complete SQL query: {:?}", e); |  | ||||||
|  |  | ||||||
|                 json!({"error": "Could not load reminders"}) |  | ||||||
|             }) |  | ||||||
|         } |  | ||||||
|         Err(e) => { |  | ||||||
|             warn!("Could not fetch channels from {}: {:?}", id, e); |  | ||||||
|  |  | ||||||
|             json!([]) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[patch("/api/guild/<id>/reminders", data = "<reminder>")] |  | ||||||
| pub async fn edit_reminder( |  | ||||||
|     id: u64, |  | ||||||
|     reminder: Json<PatchReminder>, |  | ||||||
|     serenity_context: &State<Context>, |  | ||||||
|     pool: &State<Pool<MySql>>, |  | ||||||
| ) -> JsonValue { |  | ||||||
|     let mut error = vec![]; |  | ||||||
|  |  | ||||||
|     update_field!(pool.inner(), error, reminder.[ |  | ||||||
|         attachment, |  | ||||||
|         attachment_name, |  | ||||||
|         avatar, |  | ||||||
|         content, |  | ||||||
|         embed_author, |  | ||||||
|         embed_author_url, |  | ||||||
|         embed_color, |  | ||||||
|         embed_description, |  | ||||||
|         embed_footer, |  | ||||||
|         embed_footer_url, |  | ||||||
|         embed_image_url, |  | ||||||
|         embed_thumbnail_url, |  | ||||||
|         embed_title, |  | ||||||
|         embed_fields, |  | ||||||
|         enabled, |  | ||||||
|         expires, |  | ||||||
|         interval_seconds, |  | ||||||
|         interval_months, |  | ||||||
|         name, |  | ||||||
|         pin, |  | ||||||
|         restartable, |  | ||||||
|         tts, |  | ||||||
|         username, |  | ||||||
|         utc_time |  | ||||||
|     ]); |  | ||||||
|  |  | ||||||
|     if reminder.channel > 0 { |  | ||||||
|         let channel = ChannelId(reminder.channel).to_channel_cached(&serenity_context.inner()); |  | ||||||
|         match channel { |  | ||||||
|             Some(channel) => { |  | ||||||
|                 let channel_matches_guild = channel.guild().map_or(false, |c| c.guild_id.0 == id); |  | ||||||
|  |  | ||||||
|                 if !channel_matches_guild { |  | ||||||
|                     warn!( |  | ||||||
|                         "Error in `edit_reminder`: channel {:?} not found for guild {}", |  | ||||||
|                         reminder.channel, id |  | ||||||
|                     ); |  | ||||||
|  |  | ||||||
|                     return json!({"error": "Channel not found"}); |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 let channel = create_database_channel( |  | ||||||
|                     serenity_context.inner(), |  | ||||||
|                     ChannelId(reminder.channel), |  | ||||||
|                     pool.inner(), |  | ||||||
|                 ) |  | ||||||
|                 .await; |  | ||||||
|  |  | ||||||
|                 if let Err(e) = channel { |  | ||||||
|                     warn!("`create_database_channel` returned an error code: {:?}", e); |  | ||||||
|  |  | ||||||
|                     return json!({"error": "Failed to configure channel for reminders. Please check the bot permissions"}); |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 let channel = channel.unwrap(); |  | ||||||
|  |  | ||||||
|                 match sqlx::query!( |  | ||||||
|                     "UPDATE reminders SET channel_id = ? WHERE uid = ?", |  | ||||||
|                     channel, |  | ||||||
|                     reminder.uid |  | ||||||
|                 ) |  | ||||||
|                 .execute(pool.inner()) |  | ||||||
|                 .await |  | ||||||
|                 { |  | ||||||
|                     Ok(_) => {} |  | ||||||
|                     Err(e) => { |  | ||||||
|                         warn!("Error setting channel: {:?}", e); |  | ||||||
|  |  | ||||||
|                         error.push("Couldn't set channel".to_string()) |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             None => { |  | ||||||
|                 warn!( |  | ||||||
|                     "Error in `edit_reminder`: channel {:?} not found for guild {}", |  | ||||||
|                     reminder.channel, id |  | ||||||
|                 ); |  | ||||||
|  |  | ||||||
|                 return json!({"error": "Channel not found"}); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     match sqlx::query_as_unchecked!( |  | ||||||
|         Reminder, |  | ||||||
|         "SELECT reminders.attachment, |  | ||||||
|          reminders.attachment_name, |  | ||||||
|          reminders.avatar, |  | ||||||
|          channels.channel, |  | ||||||
|          reminders.content, |  | ||||||
|          reminders.embed_author, |  | ||||||
|          reminders.embed_author_url, |  | ||||||
|          reminders.embed_color, |  | ||||||
|          reminders.embed_description, |  | ||||||
|          reminders.embed_footer, |  | ||||||
|          reminders.embed_footer_url, |  | ||||||
|          reminders.embed_image_url, |  | ||||||
|          reminders.embed_thumbnail_url, |  | ||||||
|          reminders.embed_title, |  | ||||||
|          reminders.embed_fields, |  | ||||||
|          reminders.enabled, |  | ||||||
|          reminders.expires, |  | ||||||
|          reminders.interval_seconds, |  | ||||||
|          reminders.interval_months, |  | ||||||
|          reminders.name, |  | ||||||
|          reminders.pin, |  | ||||||
|          reminders.restartable, |  | ||||||
|          reminders.tts, |  | ||||||
|          reminders.uid, |  | ||||||
|          reminders.username, |  | ||||||
|          reminders.utc_time |  | ||||||
|         FROM reminders |  | ||||||
|         LEFT JOIN channels ON channels.id = reminders.channel_id |  | ||||||
|         WHERE uid = ?", |  | ||||||
|         reminder.uid |  | ||||||
|     ) |  | ||||||
|     .fetch_one(pool.inner()) |  | ||||||
|     .await |  | ||||||
|     { |  | ||||||
|         Ok(reminder) => json!({"reminder": reminder, "errors": error}), |  | ||||||
|  |  | ||||||
|         Err(e) => { |  | ||||||
|             warn!("Error exiting `edit_reminder': {:?}", e); |  | ||||||
|  |  | ||||||
|             json!({"reminder": Option::<Reminder>::None, "errors": vec!["Unknown error"]}) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[delete("/api/guild/<_>/reminders", data = "<reminder>")] |  | ||||||
| pub async fn delete_reminder( |  | ||||||
|     reminder: Json<DeleteReminder>, |  | ||||||
|     pool: &State<Pool<MySql>>, |  | ||||||
| ) -> JsonValue { |  | ||||||
|     match sqlx::query!("DELETE FROM reminders WHERE uid = ?", reminder.uid) |  | ||||||
|         .execute(pool.inner()) |  | ||||||
|         .await |  | ||||||
|     { |  | ||||||
|         Ok(_) => { |  | ||||||
|             json!({}) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         Err(e) => { |  | ||||||
|             warn!("Error in `delete_reminder`: {:?}", e); |  | ||||||
|  |  | ||||||
|             json!({"error": "Could not delete reminder"}) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,20 +1,36 @@ | |||||||
| use std::collections::HashMap; | use std::path::Path; | ||||||
|  |  | ||||||
| use chrono::naive::NaiveDateTime; | use chrono::{naive::NaiveDateTime, Utc}; | ||||||
| use rand::{rngs::OsRng, seq::IteratorRandom}; | use rand::{rngs::OsRng, seq::IteratorRandom}; | ||||||
| use rocket::{http::CookieJar, response::Redirect}; | use rocket::{ | ||||||
| use rocket_dyn_templates::Template; |     fs::{relative, NamedFile}, | ||||||
| use serde::{Deserialize, Serialize}; |     http::CookieJar, | ||||||
| use serenity::{http::Http, model::id::ChannelId}; |     response::Redirect, | ||||||
| use sqlx::{types::Json, Executor}; |     serde::json::json, | ||||||
|  | }; | ||||||
|  | use serde::{Deserialize, Deserializer, Serialize}; | ||||||
|  | use serenity::{ | ||||||
|  |     client::Context, | ||||||
|  |     http::Http, | ||||||
|  |     model::id::{ChannelId, GuildId, UserId}, | ||||||
|  | }; | ||||||
|  | use sqlx::types::Json; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
|     consts::{CHARACTERS, DEFAULT_AVATAR}, |     check_guild_subscription, check_subscription, | ||||||
|     Database, Error, |     consts::{ | ||||||
|  |         CHARACTERS, DAY, DEFAULT_AVATAR, MAX_CONTENT_LENGTH, MAX_EMBED_AUTHOR_LENGTH, | ||||||
|  |         MAX_EMBED_DESCRIPTION_LENGTH, MAX_EMBED_FIELDS, MAX_EMBED_FIELD_TITLE_LENGTH, | ||||||
|  |         MAX_EMBED_FIELD_VALUE_LENGTH, MAX_EMBED_FOOTER_LENGTH, MAX_EMBED_TITLE_LENGTH, | ||||||
|  |         MAX_NAME_LENGTH, MAX_URL_LENGTH, MAX_USERNAME_LENGTH, MIN_INTERVAL, | ||||||
|  |     }, | ||||||
|  |     guards::transaction::Transaction, | ||||||
|  |     routes::JsonResult, | ||||||
|  |     Error, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| pub mod guild; | pub mod api; | ||||||
| pub mod user; | pub mod export; | ||||||
|  |  | ||||||
| type Unset<T> = Option<T>; | type Unset<T> = Option<T>; | ||||||
|  |  | ||||||
| @@ -34,6 +50,18 @@ fn id_default() -> u32 { | |||||||
|     0 |     0 | ||||||
| } | } | ||||||
|  |  | ||||||
|  | fn interval_default() -> Unset<Option<u32>> { | ||||||
|  |     None | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn deserialize_optional_field<'de, T, D>(deserializer: D) -> Result<Option<Option<T>>, D::Error> | ||||||
|  | where | ||||||
|  |     D: Deserializer<'de>, | ||||||
|  |     T: Deserialize<'de>, | ||||||
|  | { | ||||||
|  |     Ok(Some(Option::deserialize(deserializer)?)) | ||||||
|  | } | ||||||
|  |  | ||||||
| #[derive(Serialize, Deserialize)] | #[derive(Serialize, Deserialize)] | ||||||
| pub struct ReminderTemplate { | pub struct ReminderTemplate { | ||||||
|     #[serde(default = "id_default")] |     #[serde(default = "id_default")] | ||||||
| @@ -56,6 +84,34 @@ pub struct ReminderTemplate { | |||||||
|     embed_thumbnail_url: Option<String>, |     embed_thumbnail_url: Option<String>, | ||||||
|     embed_title: String, |     embed_title: String, | ||||||
|     embed_fields: Option<Json<Vec<EmbedField>>>, |     embed_fields: Option<Json<Vec<EmbedField>>>, | ||||||
|  |     interval_seconds: Option<u32>, | ||||||
|  |     interval_days: Option<u32>, | ||||||
|  |     interval_months: Option<u32>, | ||||||
|  |     tts: bool, | ||||||
|  |     username: Option<String>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Serialize, Deserialize)] | ||||||
|  | pub struct ReminderTemplateCsv { | ||||||
|  |     #[serde(default = "template_name_default")] | ||||||
|  |     name: String, | ||||||
|  |     attachment: Option<Vec<u8>>, | ||||||
|  |     attachment_name: Option<String>, | ||||||
|  |     avatar: Option<String>, | ||||||
|  |     content: String, | ||||||
|  |     embed_author: String, | ||||||
|  |     embed_author_url: Option<String>, | ||||||
|  |     embed_color: u32, | ||||||
|  |     embed_description: String, | ||||||
|  |     embed_footer: String, | ||||||
|  |     embed_footer_url: Option<String>, | ||||||
|  |     embed_image_url: Option<String>, | ||||||
|  |     embed_thumbnail_url: Option<String>, | ||||||
|  |     embed_title: String, | ||||||
|  |     embed_fields: Option<String>, | ||||||
|  |     interval_seconds: Option<u32>, | ||||||
|  |     interval_days: Option<u32>, | ||||||
|  |     interval_months: Option<u32>, | ||||||
|     tts: bool, |     tts: bool, | ||||||
|     username: Option<String>, |     username: Option<String>, | ||||||
| } | } | ||||||
| @@ -94,10 +150,10 @@ pub struct Reminder { | |||||||
|     enabled: bool, |     enabled: bool, | ||||||
|     expires: Option<NaiveDateTime>, |     expires: Option<NaiveDateTime>, | ||||||
|     interval_seconds: Option<u32>, |     interval_seconds: Option<u32>, | ||||||
|  |     interval_days: Option<u32>, | ||||||
|     interval_months: Option<u32>, |     interval_months: Option<u32>, | ||||||
|     #[serde(default = "name_default")] |     #[serde(default = "name_default")] | ||||||
|     name: String, |     name: String, | ||||||
|     pin: bool, |  | ||||||
|     restartable: bool, |     restartable: bool, | ||||||
|     tts: bool, |     tts: bool, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
| @@ -106,14 +162,48 @@ pub struct Reminder { | |||||||
|     utc_time: NaiveDateTime, |     utc_time: NaiveDateTime, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | #[derive(Serialize, Deserialize)] | ||||||
|  | pub struct ReminderCsv { | ||||||
|  |     #[serde(with = "base64s")] | ||||||
|  |     attachment: Option<Vec<u8>>, | ||||||
|  |     attachment_name: Option<String>, | ||||||
|  |     avatar: Option<String>, | ||||||
|  |     channel: String, | ||||||
|  |     content: String, | ||||||
|  |     embed_author: String, | ||||||
|  |     embed_author_url: Option<String>, | ||||||
|  |     embed_color: u32, | ||||||
|  |     embed_description: String, | ||||||
|  |     embed_footer: String, | ||||||
|  |     embed_footer_url: Option<String>, | ||||||
|  |     embed_image_url: Option<String>, | ||||||
|  |     embed_thumbnail_url: Option<String>, | ||||||
|  |     embed_title: String, | ||||||
|  |     embed_fields: Option<String>, | ||||||
|  |     enabled: bool, | ||||||
|  |     expires: Option<NaiveDateTime>, | ||||||
|  |     interval_seconds: Option<u32>, | ||||||
|  |     interval_days: Option<u32>, | ||||||
|  |     interval_months: Option<u32>, | ||||||
|  |     #[serde(default = "name_default")] | ||||||
|  |     name: String, | ||||||
|  |     restartable: bool, | ||||||
|  |     tts: bool, | ||||||
|  |     username: Option<String>, | ||||||
|  |     utc_time: NaiveDateTime, | ||||||
|  | } | ||||||
|  |  | ||||||
| #[derive(Deserialize)] | #[derive(Deserialize)] | ||||||
| pub struct PatchReminder { | pub struct PatchReminder { | ||||||
|     uid: String, |     uid: String, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|  |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     attachment: Unset<Option<String>>, |     attachment: Unset<Option<String>>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|  |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     attachment_name: Unset<Option<String>>, |     attachment_name: Unset<Option<String>>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|  |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     avatar: Unset<Option<String>>, |     avatar: Unset<Option<String>>, | ||||||
|     #[serde(default = "channel_default")] |     #[serde(default = "channel_default")] | ||||||
|     #[serde(with = "string")] |     #[serde(with = "string")] | ||||||
| @@ -123,6 +213,7 @@ pub struct PatchReminder { | |||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|     embed_author: Unset<String>, |     embed_author: Unset<String>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|  |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     embed_author_url: Unset<Option<String>>, |     embed_author_url: Unset<Option<String>>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|     embed_color: Unset<u32>, |     embed_color: Unset<u32>, | ||||||
| @@ -131,10 +222,13 @@ pub struct PatchReminder { | |||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|     embed_footer: Unset<String>, |     embed_footer: Unset<String>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|  |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     embed_footer_url: Unset<Option<String>>, |     embed_footer_url: Unset<Option<String>>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|  |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     embed_image_url: Unset<Option<String>>, |     embed_image_url: Unset<Option<String>>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|  |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     embed_thumbnail_url: Unset<Option<String>>, |     embed_thumbnail_url: Unset<Option<String>>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|     embed_title: Unset<String>, |     embed_title: Unset<String>, | ||||||
| @@ -143,25 +237,54 @@ pub struct PatchReminder { | |||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|     enabled: Unset<bool>, |     enabled: Unset<bool>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|  |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     expires: Unset<Option<NaiveDateTime>>, |     expires: Unset<Option<NaiveDateTime>>, | ||||||
|     #[serde(default)] |     #[serde(default = "interval_default")] | ||||||
|  |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     interval_seconds: Unset<Option<u32>>, |     interval_seconds: Unset<Option<u32>>, | ||||||
|     #[serde(default)] |     #[serde(default = "interval_default")] | ||||||
|  |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|  |     interval_days: Unset<Option<u32>>, | ||||||
|  |     #[serde(default = "interval_default")] | ||||||
|  |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     interval_months: Unset<Option<u32>>, |     interval_months: Unset<Option<u32>>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|     name: Unset<String>, |     name: Unset<String>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|     pin: Unset<bool>, |  | ||||||
|     #[serde(default)] |  | ||||||
|     restartable: Unset<bool>, |     restartable: Unset<bool>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|     tts: Unset<bool>, |     tts: Unset<bool>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|  |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     username: Unset<Option<String>>, |     username: Unset<Option<String>>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|     utc_time: Unset<NaiveDateTime>, |     utc_time: Unset<NaiveDateTime>, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | impl PatchReminder { | ||||||
|  |     fn message_ok(&self) -> bool { | ||||||
|  |         self.content.as_ref().map_or(true, |c| c.len() <= MAX_CONTENT_LENGTH) | ||||||
|  |             && self.embed_author.as_ref().map_or(true, |c| c.len() <= MAX_EMBED_AUTHOR_LENGTH) | ||||||
|  |             && self | ||||||
|  |                 .embed_description | ||||||
|  |                 .as_ref() | ||||||
|  |                 .map_or(true, |c| c.len() <= MAX_EMBED_DESCRIPTION_LENGTH) | ||||||
|  |             && self.embed_footer.as_ref().map_or(true, |c| c.len() <= MAX_EMBED_FOOTER_LENGTH) | ||||||
|  |             && self.embed_title.as_ref().map_or(true, |c| c.len() <= MAX_EMBED_TITLE_LENGTH) | ||||||
|  |             && self.embed_fields.as_ref().map_or(true, |c| { | ||||||
|  |                 c.0.len() <= MAX_EMBED_FIELDS | ||||||
|  |                     && c.0.iter().all(|f| { | ||||||
|  |                         f.title.len() <= MAX_EMBED_FIELD_TITLE_LENGTH | ||||||
|  |                             && f.value.len() <= MAX_EMBED_FIELD_VALUE_LENGTH | ||||||
|  |                     }) | ||||||
|  |             }) | ||||||
|  |             && self | ||||||
|  |                 .username | ||||||
|  |                 .as_ref() | ||||||
|  |                 .map_or(true, |c| c.as_ref().map_or(true, |v| v.len() <= MAX_USERNAME_LENGTH)) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| pub fn generate_uid() -> String { | pub fn generate_uid() -> String { | ||||||
|     let mut generator: OsRng = Default::default(); |     let mut generator: OsRng = Default::default(); | ||||||
|  |  | ||||||
| @@ -213,8 +336,8 @@ mod base64s { | |||||||
|     where |     where | ||||||
|         D: Deserializer<'de>, |         D: Deserializer<'de>, | ||||||
|     { |     { | ||||||
|         let string = String::deserialize(deserializer)?; |         let string = Option::<String>::deserialize(deserializer)?; | ||||||
|         Some(base64::decode(string).map_err(de::Error::custom)).transpose() |         Some(string.map(|b| base64::decode(b).map_err(de::Error::custom))).flatten().transpose() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -223,16 +346,259 @@ pub struct DeleteReminder { | |||||||
|     uid: String, |     uid: String, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | #[derive(Deserialize)] | ||||||
|  | pub struct ImportBody { | ||||||
|  |     body: String, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Serialize, Deserialize)] | ||||||
|  | pub struct TodoCsv { | ||||||
|  |     value: String, | ||||||
|  |     channel_id: Option<String>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | pub(crate) async fn create_reminder( | ||||||
|  |     ctx: &Context, | ||||||
|  |     transaction: &mut Transaction<'_>, | ||||||
|  |     guild_id: GuildId, | ||||||
|  |     user_id: UserId, | ||||||
|  |     reminder: Reminder, | ||||||
|  | ) -> JsonResult { | ||||||
|  |     // check guild in db | ||||||
|  |     match sqlx::query!("SELECT 1 as A FROM guilds WHERE guild = ?", guild_id.0) | ||||||
|  |         .fetch_one(transaction.executor()) | ||||||
|  |         .await | ||||||
|  |     { | ||||||
|  |         Err(sqlx::Error::RowNotFound) => { | ||||||
|  |             if sqlx::query!("INSERT INTO guilds (guild) VALUES (?)", guild_id.0) | ||||||
|  |                 .execute(transaction.executor()) | ||||||
|  |                 .await | ||||||
|  |                 .is_err() | ||||||
|  |             { | ||||||
|  |                 return Err(json!({"error": "Guild could not be created"})); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         _ => {} | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // validate channel | ||||||
|  |     let channel = ChannelId(reminder.channel).to_channel_cached(&ctx); | ||||||
|  |     let channel_exists = channel.is_some(); | ||||||
|  |  | ||||||
|  |     let channel_matches_guild = | ||||||
|  |         channel.map_or(false, |c| c.guild().map_or(false, |c| c.guild_id == guild_id)); | ||||||
|  |  | ||||||
|  |     if !channel_matches_guild || !channel_exists { | ||||||
|  |         warn!( | ||||||
|  |             "Error in `create_reminder`: channel {} not found for guild {} (channel exists: {})", | ||||||
|  |             reminder.channel, guild_id, channel_exists | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         return Err(json!({"error": "Channel not found"})); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let channel = create_database_channel(&ctx, ChannelId(reminder.channel), transaction).await; | ||||||
|  |  | ||||||
|  |     if let Err(e) = channel { | ||||||
|  |         warn!("`create_database_channel` returned an error code: {:?}", e); | ||||||
|  |  | ||||||
|  |         return Err( | ||||||
|  |             json!({"error": "Failed to configure channel for reminders. Please check the bot permissions"}), | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let channel = channel.unwrap(); | ||||||
|  |  | ||||||
|  |     // validate lengths | ||||||
|  |     check_length!(MAX_NAME_LENGTH, reminder.name); | ||||||
|  |     check_length!(MAX_CONTENT_LENGTH, reminder.content); | ||||||
|  |     check_length!(MAX_EMBED_DESCRIPTION_LENGTH, reminder.embed_description); | ||||||
|  |     check_length!(MAX_EMBED_TITLE_LENGTH, reminder.embed_title); | ||||||
|  |     check_length!(MAX_EMBED_AUTHOR_LENGTH, reminder.embed_author); | ||||||
|  |     check_length!(MAX_EMBED_FOOTER_LENGTH, reminder.embed_footer); | ||||||
|  |     check_length_opt!(MAX_EMBED_FIELDS, reminder.embed_fields); | ||||||
|  |     if let Some(fields) = &reminder.embed_fields { | ||||||
|  |         for field in &fields.0 { | ||||||
|  |             check_length!(MAX_EMBED_FIELD_VALUE_LENGTH, field.value); | ||||||
|  |             check_length!(MAX_EMBED_FIELD_TITLE_LENGTH, field.title); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     check_length_opt!(MAX_USERNAME_LENGTH, reminder.username); | ||||||
|  |     check_length_opt!( | ||||||
|  |         MAX_URL_LENGTH, | ||||||
|  |         reminder.embed_footer_url, | ||||||
|  |         reminder.embed_thumbnail_url, | ||||||
|  |         reminder.embed_author_url, | ||||||
|  |         reminder.embed_image_url, | ||||||
|  |         reminder.avatar | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     // validate urls | ||||||
|  |     check_url_opt!( | ||||||
|  |         reminder.embed_footer_url, | ||||||
|  |         reminder.embed_thumbnail_url, | ||||||
|  |         reminder.embed_author_url, | ||||||
|  |         reminder.embed_image_url, | ||||||
|  |         reminder.avatar | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     // validate time and interval | ||||||
|  |     if reminder.utc_time < Utc::now().naive_utc() { | ||||||
|  |         return Err(json!({"error": "Time must be in the future"})); | ||||||
|  |     } | ||||||
|  |     if reminder.interval_seconds.is_some() | ||||||
|  |         || reminder.interval_days.is_some() | ||||||
|  |         || reminder.interval_months.is_some() | ||||||
|  |     { | ||||||
|  |         if reminder.interval_months.unwrap_or(0) * 30 * DAY as u32 | ||||||
|  |             + reminder.interval_days.unwrap_or(0) * DAY as u32 | ||||||
|  |             + reminder.interval_seconds.unwrap_or(0) | ||||||
|  |             < *MIN_INTERVAL | ||||||
|  |         { | ||||||
|  |             return Err(json!({"error": "Interval too short"})); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // check patreon if necessary | ||||||
|  |     if reminder.interval_seconds.is_some() | ||||||
|  |         || reminder.interval_days.is_some() | ||||||
|  |         || reminder.interval_months.is_some() | ||||||
|  |     { | ||||||
|  |         if !check_guild_subscription(&ctx, guild_id).await | ||||||
|  |             && !check_subscription(&ctx, user_id).await | ||||||
|  |         { | ||||||
|  |             return Err(json!({"error": "Patreon is required to set intervals"})); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let name = if reminder.name.is_empty() { name_default() } else { reminder.name.clone() }; | ||||||
|  |     let username = if reminder.username.as_ref().map(|s| s.is_empty()).unwrap_or(true) { | ||||||
|  |         None | ||||||
|  |     } else { | ||||||
|  |         reminder.username | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     let new_uid = generate_uid(); | ||||||
|  |  | ||||||
|  |     // write to db | ||||||
|  |     match sqlx::query!( | ||||||
|  |         "INSERT INTO reminders ( | ||||||
|  |          uid, | ||||||
|  |          attachment, | ||||||
|  |          attachment_name, | ||||||
|  |          channel_id, | ||||||
|  |          avatar, | ||||||
|  |          content, | ||||||
|  |          embed_author, | ||||||
|  |          embed_author_url, | ||||||
|  |          embed_color, | ||||||
|  |          embed_description, | ||||||
|  |          embed_footer, | ||||||
|  |          embed_footer_url, | ||||||
|  |          embed_image_url, | ||||||
|  |          embed_thumbnail_url, | ||||||
|  |          embed_title, | ||||||
|  |          embed_fields, | ||||||
|  |          enabled, | ||||||
|  |          expires, | ||||||
|  |          interval_seconds, | ||||||
|  |          interval_days, | ||||||
|  |          interval_months, | ||||||
|  |          name, | ||||||
|  |          restartable, | ||||||
|  |          tts, | ||||||
|  |          username, | ||||||
|  |          `utc_time` | ||||||
|  |         ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)", | ||||||
|  |         new_uid, | ||||||
|  |         reminder.attachment, | ||||||
|  |         reminder.attachment_name, | ||||||
|  |         channel, | ||||||
|  |         reminder.avatar, | ||||||
|  |         reminder.content, | ||||||
|  |         reminder.embed_author, | ||||||
|  |         reminder.embed_author_url, | ||||||
|  |         reminder.embed_color, | ||||||
|  |         reminder.embed_description, | ||||||
|  |         reminder.embed_footer, | ||||||
|  |         reminder.embed_footer_url, | ||||||
|  |         reminder.embed_image_url, | ||||||
|  |         reminder.embed_thumbnail_url, | ||||||
|  |         reminder.embed_title, | ||||||
|  |         reminder.embed_fields, | ||||||
|  |         reminder.enabled, | ||||||
|  |         reminder.expires, | ||||||
|  |         reminder.interval_seconds, | ||||||
|  |         reminder.interval_days, | ||||||
|  |         reminder.interval_months, | ||||||
|  |         name, | ||||||
|  |         reminder.restartable, | ||||||
|  |         reminder.tts, | ||||||
|  |         username, | ||||||
|  |         reminder.utc_time, | ||||||
|  |     ) | ||||||
|  |     .execute(transaction.executor()) | ||||||
|  |     .await | ||||||
|  |     { | ||||||
|  |         Ok(_) => sqlx::query_as_unchecked!( | ||||||
|  |             Reminder, | ||||||
|  |             "SELECT | ||||||
|  |              reminders.attachment, | ||||||
|  |              reminders.attachment_name, | ||||||
|  |              reminders.avatar, | ||||||
|  |              channels.channel, | ||||||
|  |              reminders.content, | ||||||
|  |              reminders.embed_author, | ||||||
|  |              reminders.embed_author_url, | ||||||
|  |              reminders.embed_color, | ||||||
|  |              reminders.embed_description, | ||||||
|  |              reminders.embed_footer, | ||||||
|  |              reminders.embed_footer_url, | ||||||
|  |              reminders.embed_image_url, | ||||||
|  |              reminders.embed_thumbnail_url, | ||||||
|  |              reminders.embed_title, | ||||||
|  |              reminders.embed_fields, | ||||||
|  |              reminders.enabled, | ||||||
|  |              reminders.expires, | ||||||
|  |              reminders.interval_seconds, | ||||||
|  |              reminders.interval_days, | ||||||
|  |              reminders.interval_months, | ||||||
|  |              reminders.name, | ||||||
|  |              reminders.restartable, | ||||||
|  |              reminders.tts, | ||||||
|  |              reminders.uid, | ||||||
|  |              reminders.username, | ||||||
|  |              reminders.utc_time | ||||||
|  |             FROM reminders | ||||||
|  |             LEFT JOIN channels ON channels.id = reminders.channel_id | ||||||
|  |             WHERE uid = ?", | ||||||
|  |             new_uid | ||||||
|  |         ) | ||||||
|  |         .fetch_one(transaction.executor()) | ||||||
|  |         .await | ||||||
|  |         .map(|r| Ok(json!(r))) | ||||||
|  |         .unwrap_or_else(|e| { | ||||||
|  |             warn!("Failed to complete SQL query: {:?}", e); | ||||||
|  |  | ||||||
|  |             Err(json!({"error": "Could not load reminder"})) | ||||||
|  |         }), | ||||||
|  |  | ||||||
|  |         Err(e) => { | ||||||
|  |             warn!("Error in `create_reminder`: Could not execute query: {:?}", e); | ||||||
|  |  | ||||||
|  |             Err(json!({"error": "Unknown error"})) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| async fn create_database_channel( | async fn create_database_channel( | ||||||
|     ctx: impl AsRef<Http>, |     ctx: impl AsRef<Http>, | ||||||
|     channel: ChannelId, |     channel: ChannelId, | ||||||
|     pool: impl Executor<'_, Database = Database> + Copy, |     transaction: &mut Transaction<'_>, | ||||||
| ) -> Result<u32, crate::Error> { | ) -> Result<u32, crate::Error> { | ||||||
|     println!("{:?}", channel); |  | ||||||
|  |  | ||||||
|     let row = |     let row = | ||||||
|         sqlx::query!("SELECT webhook_token, webhook_id FROM channels WHERE channel = ?", channel.0) |         sqlx::query!("SELECT webhook_token, webhook_id FROM channels WHERE channel = ?", channel.0) | ||||||
|             .fetch_one(pool) |             .fetch_one(transaction.executor()) | ||||||
|             .await; |             .await; | ||||||
|  |  | ||||||
|     match row { |     match row { | ||||||
| @@ -249,7 +615,7 @@ async fn create_database_channel( | |||||||
|                     webhook.token, |                     webhook.token, | ||||||
|                     channel.0 |                     channel.0 | ||||||
|                 ) |                 ) | ||||||
|                 .execute(pool) |                 .execute(transaction.executor()) | ||||||
|                 .await |                 .await | ||||||
|                 .map_err(|e| Error::SQLx(e))?; |                 .map_err(|e| Error::SQLx(e))?; | ||||||
|             } |             } | ||||||
| @@ -275,7 +641,7 @@ async fn create_database_channel( | |||||||
|                 webhook.token, |                 webhook.token, | ||||||
|                 channel.0 |                 channel.0 | ||||||
|             ) |             ) | ||||||
|             .execute(pool) |             .execute(transaction.executor()) | ||||||
|             .await |             .await | ||||||
|             .map_err(|e| Error::SQLx(e))?; |             .map_err(|e| Error::SQLx(e))?; | ||||||
|  |  | ||||||
| @@ -286,7 +652,7 @@ async fn create_database_channel( | |||||||
|     }?; |     }?; | ||||||
|  |  | ||||||
|     let row = sqlx::query!("SELECT id FROM channels WHERE channel = ?", channel.0) |     let row = sqlx::query!("SELECT id FROM channels WHERE channel = ?", channel.0) | ||||||
|         .fetch_one(pool) |         .fetch_one(transaction.executor()) | ||||||
|         .await |         .await | ||||||
|         .map_err(|e| Error::SQLx(e))?; |         .map_err(|e| Error::SQLx(e))?; | ||||||
|  |  | ||||||
| @@ -294,20 +660,26 @@ async fn create_database_channel( | |||||||
| } | } | ||||||
|  |  | ||||||
| #[get("/")] | #[get("/")] | ||||||
| pub async fn dashboard_home(cookies: &CookieJar<'_>) -> Result<Template, Redirect> { | pub async fn dashboard_home(cookies: &CookieJar<'_>) -> Result<NamedFile, Redirect> { | ||||||
|     if cookies.get_private("userid").is_some() { |     if cookies.get_private("userid").is_some() { | ||||||
|         let map: HashMap<&str, String> = HashMap::new(); |         NamedFile::open(Path::new(relative!("static/index.html"))).await.map_err(|e| { | ||||||
|         Ok(Template::render("dashboard", &map)) |             warn!("Couldn't render dashboard: {:?}", e); | ||||||
|  |  | ||||||
|  |             Redirect::to("/login/discord") | ||||||
|  |         }) | ||||||
|     } else { |     } else { | ||||||
|         Err(Redirect::to("/login/discord")) |         Err(Redirect::to("/login/discord")) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| #[get("/<_>")] | #[get("/<_..>")] | ||||||
| pub async fn dashboard(cookies: &CookieJar<'_>) -> Result<Template, Redirect> { | pub async fn dashboard(cookies: &CookieJar<'_>) -> Result<NamedFile, Redirect> { | ||||||
|     if cookies.get_private("userid").is_some() { |     if cookies.get_private("userid").is_some() { | ||||||
|         let map: HashMap<&str, String> = HashMap::new(); |         NamedFile::open(Path::new(relative!("static/index.html"))).await.map_err(|e| { | ||||||
|         Ok(Template::render("dashboard", &map)) |             warn!("Couldn't render dashboard: {:?}", e); | ||||||
|  |  | ||||||
|  |             Redirect::to("/login/discord") | ||||||
|  |         }) | ||||||
|     } else { |     } else { | ||||||
|         Err(Redirect::to("/login/discord")) |         Err(Redirect::to("/login/discord")) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ use rocket::{ | |||||||
| }; | }; | ||||||
| use serenity::model::user::User; | use serenity::model::user::User; | ||||||
|  |  | ||||||
| use crate::consts::DISCORD_API; | use crate::{consts::DISCORD_API, routes}; | ||||||
|  |  | ||||||
| #[get("/discord")] | #[get("/discord")] | ||||||
| pub async fn discord_login( | pub async fn discord_login( | ||||||
| @@ -25,7 +25,6 @@ pub async fn discord_login( | |||||||
|         // Set the desired scopes. |         // Set the desired scopes. | ||||||
|         .add_scope(Scope::new("identify".to_string())) |         .add_scope(Scope::new("identify".to_string())) | ||||||
|         .add_scope(Scope::new("guilds".to_string())) |         .add_scope(Scope::new("guilds".to_string())) | ||||||
|         .add_scope(Scope::new("email".to_string())) |  | ||||||
|         // Set the PKCE code challenge. |         // Set the PKCE code challenge. | ||||||
|         .set_pkce_challenge(pkce_challenge) |         .set_pkce_challenge(pkce_challenge) | ||||||
|         .url(); |         .url(); | ||||||
| @@ -53,6 +52,15 @@ pub async fn discord_login( | |||||||
|     Redirect::to(auth_url.to_string()) |     Redirect::to(auth_url.to_string()) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | #[get("/discord/logout")] | ||||||
|  | pub async fn discord_logout(cookies: &CookieJar<'_>) -> Redirect { | ||||||
|  |     cookies.remove_private(Cookie::named("username")); | ||||||
|  |     cookies.remove_private(Cookie::named("userid")); | ||||||
|  |     cookies.remove_private(Cookie::named("access_token")); | ||||||
|  |  | ||||||
|  |     Redirect::to(uri!(routes::index)) | ||||||
|  | } | ||||||
|  |  | ||||||
| #[get("/discord/authorized?<code>&<state>")] | #[get("/discord/authorized?<code>&<state>")] | ||||||
| pub async fn discord_callback( | pub async fn discord_callback( | ||||||
|     code: &str, |     code: &str, | ||||||
| @@ -136,14 +144,14 @@ pub async fn discord_callback( | |||||||
|                     Err(Flash::new( |                     Err(Flash::new( | ||||||
|                         Redirect::to(uri!(super::return_to_same_site(""))), |                         Redirect::to(uri!(super::return_to_same_site(""))), | ||||||
|                         "warning", |                         "warning", | ||||||
|                         "Your login request was rejected", |                         "Your login request was rejected. The server may be misconfigured. Please retry or alert us in Discord.", | ||||||
|                     )) |                     )) | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             Err(Flash::new(Redirect::to(uri!(super::return_to_same_site(""))), "danger", "Your request failed to validate, and so has been rejected (error: CSRF Validation Failure)")) |             Err(Flash::new(Redirect::to(uri!(super::return_to_same_site(""))), "danger", "Your request failed to validate, and so has been rejected (CSRF Validation Failure)")) | ||||||
|         } |         } | ||||||
|     } else { |     } else { | ||||||
|         Err(Flash::new(Redirect::to(uri!(super::return_to_same_site(""))), "warning", "Your request was missing information, and so has been rejected (error: CSRF Validation Tokens Missing)")) |         Err(Flash::new(Redirect::to(uri!(super::return_to_same_site(""))), "warning", "Your request was missing information, and so has been rejected (CSRF Validation Tokens Missing)")) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										18
									
								
								web/src/routes/metrics.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,18 @@ | |||||||
|  | use prometheus; | ||||||
|  |  | ||||||
|  | use crate::metrics::REGISTRY; | ||||||
|  |  | ||||||
|  | #[get("/metrics")] | ||||||
|  | pub async fn metrics() -> String { | ||||||
|  |     let encoder = prometheus::TextEncoder::new(); | ||||||
|  |     let res_custom = encoder.encode_to_string(®ISTRY.gather()); | ||||||
|  |  | ||||||
|  |     match res_custom { | ||||||
|  |         Ok(s) => s, | ||||||
|  |         Err(e) => { | ||||||
|  |             warn!("Error encoding metrics: {:?}", e); | ||||||
|  |  | ||||||
|  |             String::new() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,11 +1,16 @@ | |||||||
|  | pub mod admin; | ||||||
| pub mod dashboard; | pub mod dashboard; | ||||||
| pub mod login; | pub mod login; | ||||||
|  | pub mod metrics; | ||||||
|  | pub mod report; | ||||||
|  |  | ||||||
| use std::collections::HashMap; | use std::collections::HashMap; | ||||||
|  |  | ||||||
| use rocket::request::FlashMessage; | use rocket::{request::FlashMessage, serde::json::Value as JsonValue}; | ||||||
| use rocket_dyn_templates::Template; | use rocket_dyn_templates::Template; | ||||||
|  |  | ||||||
|  | pub type JsonResult = Result<JsonValue, JsonValue>; | ||||||
|  |  | ||||||
| #[get("/")] | #[get("/")] | ||||||
| pub async fn index(flash: Option<FlashMessage<'_>>) -> Template { | pub async fn index(flash: Option<FlashMessage<'_>>) -> Template { | ||||||
|     let mut map: HashMap<&str, String> = HashMap::new(); |     let mut map: HashMap<&str, String> = HashMap::new(); | ||||||
| @@ -86,3 +91,21 @@ pub async fn help_macros() -> Template { | |||||||
|     let map: HashMap<&str, String> = HashMap::new(); |     let map: HashMap<&str, String> = HashMap::new(); | ||||||
|     Template::render("support/macros", &map) |     Template::render("support/macros", &map) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | #[get("/intervals")] | ||||||
|  | pub async fn help_intervals() -> Template { | ||||||
|  |     let map: HashMap<&str, String> = HashMap::new(); | ||||||
|  |     Template::render("support/intervals", &map) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[get("/dashboard")] | ||||||
|  | pub async fn help_dashboard() -> Template { | ||||||
|  |     let map: HashMap<&str, String> = HashMap::new(); | ||||||
|  |     Template::render("support/dashboard", &map) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[get("/iemanager")] | ||||||
|  | pub async fn help_iemanager() -> Template { | ||||||
|  |     let map: HashMap<&str, String> = HashMap::new(); | ||||||
|  |     Template::render("support/iemanager", &map) | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										48
									
								
								web/src/routes/report.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,48 @@ | |||||||
|  | use rocket::{ | ||||||
|  |     http::CookieJar, | ||||||
|  |     serde::{ | ||||||
|  |         json::{json, Json}, | ||||||
|  |         Deserialize, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | use crate::routes::JsonResult; | ||||||
|  |  | ||||||
|  | #[derive(Deserialize)] | ||||||
|  | pub struct ClientError { | ||||||
|  |     #[serde(rename = "reporterId")] | ||||||
|  |     reporter_id: String, | ||||||
|  |     url: String, | ||||||
|  |     #[serde(rename = "relativeTimestamp")] | ||||||
|  |     relative_timestamp: i64, | ||||||
|  |     #[serde(rename = "errorMessage")] | ||||||
|  |     error_message: String, | ||||||
|  |     #[serde(rename = "errorLine")] | ||||||
|  |     error_line: u64, | ||||||
|  |     #[serde(rename = "errorFile")] | ||||||
|  |     error_file: String, | ||||||
|  |     #[serde(rename = "errorType")] | ||||||
|  |     error_type: String, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[post("/report", data = "<client_error>")] | ||||||
|  | pub async fn report_error(cookies: &CookieJar<'_>, client_error: Json<ClientError>) -> JsonResult { | ||||||
|  |     if let Some(user_id) = cookies.get_private("userid") { | ||||||
|  |         error!( | ||||||
|  |             "User {} reports a client-side error. | ||||||
|  | {}, {}:{} at {}ms | ||||||
|  | {}: {} | ||||||
|  | Chain: {}", | ||||||
|  |             user_id, | ||||||
|  |             client_error.url, | ||||||
|  |             client_error.error_file, | ||||||
|  |             client_error.error_line, | ||||||
|  |             client_error.relative_timestamp, | ||||||
|  |             client_error.error_type, | ||||||
|  |             client_error.error_message, | ||||||
|  |             client_error.reporter_id | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Ok(json!({})) | ||||||
|  | } | ||||||
| @@ -11,10 +11,26 @@ div.reminderContent.is-collapsed .column.discord-frame { | |||||||
|     display: none; |     display: none; | ||||||
| } | } | ||||||
|  |  | ||||||
| div.reminderContent.is-collapsed .collapses { | div.reminderContent.is-collapsed .column.settings { | ||||||
|     display: none; |     display: none; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | div.reminderContent.is-collapsed .reminder-settings { | ||||||
|  |     margin-bottom: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderContent.is-collapsed .button-row { | ||||||
|  |     display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderContent.is-collapsed .button-row-edit { | ||||||
|  |     display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderContent.is-collapsed .reminder-topbar { | ||||||
|  |     padding-bottom: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
| div.reminderContent.is-collapsed .invert-collapses { | div.reminderContent.is-collapsed .invert-collapses { | ||||||
|     display: inline-flex; |     display: inline-flex; | ||||||
| } | } | ||||||
| @@ -23,42 +39,42 @@ div.reminderContent .invert-collapses { | |||||||
|     display: none; |     display: none; | ||||||
| } | } | ||||||
|  |  | ||||||
| div.reminderContent.is-collapsed .settings { |  | ||||||
|     display: flex; |  | ||||||
|     flex-direction: row; |  | ||||||
|     padding-bottom: 0; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| div.reminderContent.is-collapsed .channel-field { |  | ||||||
|     display: inline-flex; |  | ||||||
|     order: 1; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| div.reminderContent.is-collapsed .reminder-topbar { |  | ||||||
|     display: inline-flex; |  | ||||||
|     margin-bottom: 0px; |  | ||||||
|     flex-grow: 1; |  | ||||||
|     order: 2; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| div.reminderContent.is-collapsed input[name="name"] { | div.reminderContent.is-collapsed input[name="name"] { | ||||||
|     display: inline-flex; |     display: inline-flex; | ||||||
|     flex-grow: 1; |     flex-grow: 1; | ||||||
|     border: none; |     border: none; | ||||||
|     font-weight: 700; |  | ||||||
|     background: none; |     background: none; | ||||||
|  |     box-shadow: none; | ||||||
|  |     opacity: 1; | ||||||
| } | } | ||||||
|  |  | ||||||
| div.reminderContent.is-collapsed button.hide-box { | div.reminderContent.is-collapsed .hide-box { | ||||||
|     display: inline-flex; |     display: inline-flex; | ||||||
| } | } | ||||||
|  |  | ||||||
| div.reminderContent.is-collapsed button.hide-box i { | div.reminderContent.is-collapsed .hide-box i { | ||||||
|     transform: rotate(90deg); |     transform: rotate(90deg); | ||||||
| } | } | ||||||
| /* END */ | /* END */ | ||||||
|  |  | ||||||
| /* dashboard styles */ | /* dashboard styles */ | ||||||
|  | .hide-box { | ||||||
|  |     border: none; | ||||||
|  |     background: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .hide-box:focus { | ||||||
|  |     outline: none; | ||||||
|  |     box-shadow: none !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .channel-bar { | ||||||
|  |     display: flex; | ||||||
|  |     justify-content: center; | ||||||
|  |     flex-direction: column; | ||||||
|  |     font-weight: bold; | ||||||
|  | } | ||||||
|  |  | ||||||
| button.inline-btn { | button.inline-btn { | ||||||
|     height: 100%; |     height: 100%; | ||||||
|     padding: 5px; |     padding: 5px; | ||||||
| @@ -85,18 +101,86 @@ div.discord-embed { | |||||||
|     position: relative; |     position: relative; | ||||||
| } | } | ||||||
|  |  | ||||||
| div.reminderContent { | div.split-controls { | ||||||
|     padding: 2px; |     display: flex; | ||||||
|     background-color: #f5f5f5; |     flex-direction: column; | ||||||
|     border-radius: 8px; |     justify-content: space-between; | ||||||
|     margin: 8px; |     flex-grow: 2; | ||||||
| } | } | ||||||
|  |  | ||||||
| div.interval-group > button { | .reminder-topbar > div { | ||||||
|     margin-left: auto; |     padding-left: 6px; | ||||||
|  |     padding-right: 6px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .settings { | ||||||
|  |     display: flex; | ||||||
|  |     flex-direction: column; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .name-bar { | ||||||
|  |     flex-grow: 1; | ||||||
|  |     flex-shrink: 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .hide-button-bar { | ||||||
|  |     flex-grow: 0; | ||||||
|  |     flex-shrink: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .patreon-only { | ||||||
|  |     padding-bottom: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .tts-row { | ||||||
|  |     padding-bottom: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .reminder-topbar { | ||||||
|  |     display: flex; | ||||||
|  |     margin-bottom: 0 !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .reminder-settings { | ||||||
|  |     margin-top: 0 !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .reminder-settings > .column { | ||||||
|  |     flex-grow: 0; | ||||||
|  |     flex-shrink: 0; | ||||||
|  |     flex-basis: 50%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderContent { | ||||||
|  |     margin-top: 10px; | ||||||
|  |     margin-bottom: 10px; | ||||||
|  |     padding: 14px; | ||||||
|  |     background-color: #f5f5f5; | ||||||
|  |     border-radius: 8px; | ||||||
| } | } | ||||||
|  |  | ||||||
| /* Interval inputs */ | /* Interval inputs */ | ||||||
|  | div.interval-group { | ||||||
|  |     height: unset !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.interval-group .clear:focus { | ||||||
|  |     outline: none; | ||||||
|  |     box-shadow: none !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.interval-group .no-break { | ||||||
|  |     text-wrap: avoid; | ||||||
|  |     white-space: nowrap; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.interval-group .clear { | ||||||
|  |     border: none; | ||||||
|  |     background: none; | ||||||
|  |     padding: 1px; | ||||||
|  |     margin-right: -3px; | ||||||
|  | } | ||||||
|  |  | ||||||
| div.interval-group > .interval-group-left input { | div.interval-group > .interval-group-left input { | ||||||
|     -webkit-appearance: none; |     -webkit-appearance: none; | ||||||
|     border-style: none; |     border-style: none; | ||||||
| @@ -110,12 +194,13 @@ div.interval-group > .interval-group-left input.w2 { | |||||||
| } | } | ||||||
|  |  | ||||||
| div.interval-group > .interval-group-left input.w3 { | div.interval-group > .interval-group-left input.w3 { | ||||||
|     width: 6ch; |     width: 3ch; | ||||||
| } | } | ||||||
|  |  | ||||||
| div.interval-group { | div.interval-group { | ||||||
|     display: flex; |     display: flex; | ||||||
|     flex-direction: row; |     flex-direction: row; | ||||||
|  |     justify-content: space-between; | ||||||
| } | } | ||||||
| /* !Interval inputs */ | /* !Interval inputs */ | ||||||
|  |  | ||||||
| @@ -133,17 +218,16 @@ div.inset-content { | |||||||
|     margin-right: 10%; |     margin-right: 10%; | ||||||
| } | } | ||||||
|  |  | ||||||
| div.flash-message { | div.flash-container { | ||||||
|     position: fixed; |     position: fixed; | ||||||
|  |     width: 100%; | ||||||
|  |     bottom: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.flash-message { | ||||||
|     width: calc(100% - 32px); |     width: calc(100% - 32px); | ||||||
|     margin: 16px !important; |     margin: 16px !important; | ||||||
|     z-index: 99; |     z-index: 99; | ||||||
|     bottom: 0; |  | ||||||
|     display: none; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| div.flash-message.is-active { |  | ||||||
|     display: block; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| body { | body { | ||||||
| @@ -180,6 +264,23 @@ div#pageNavbar a { | |||||||
|     text-align: center; |     text-align: center; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .navbar-burger { | ||||||
|  |     flex-shrink: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .navbar-item.pageTitle { | ||||||
|  |     flex-shrink: 1; | ||||||
|  |     white-space: nowrap; | ||||||
|  |     overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .dashboard-burger, .dashboard-burger:active, .dashboard-burger.is-active { | ||||||
|  |     background-color: #adc99c !important; | ||||||
|  |     border-radius: 14px; | ||||||
|  |     padding: 6px; | ||||||
|  |     background-clip: content-box; | ||||||
|  | } | ||||||
|  |  | ||||||
| div#pageNavbar a:hover { | div#pageNavbar a:hover { | ||||||
|     background-color: #4a4a4a; |     background-color: #4a4a4a; | ||||||
| } | } | ||||||
| @@ -206,17 +307,24 @@ div.dashboard-sidebar { | |||||||
|     padding-right: 0; |     padding-right: 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| div.dashboard-sidebar:not(.mobile-sidebar) { | ul.guildList { | ||||||
|     display: flex; |     flex-grow: 1; | ||||||
|     flex-direction: column; |     flex-shrink: 1; | ||||||
|  |     overflow: auto; | ||||||
| } | } | ||||||
|  |  | ||||||
| div.dashboard-sidebar:not(.mobile-sidebar) .aside-footer { | div.dashboard-sidebar:not(.mobile-sidebar) .aside-footer { | ||||||
|  |     flex-shrink: 0; | ||||||
|  |     flex-grow: 0; | ||||||
|     position: fixed; |     position: fixed; | ||||||
|     bottom: 0; |     bottom: 0; | ||||||
|     width: 226px; |     width: 226px; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | div.dashboard-sidebar svg { | ||||||
|  |     flex-shrink: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
| div.mobile-sidebar { | div.mobile-sidebar { | ||||||
|     z-index: 100; |     z-index: 100; | ||||||
|     min-height: 100vh; |     min-height: 100vh; | ||||||
| @@ -288,11 +396,12 @@ textarea, input { | |||||||
|     width: 100%; |     width: 100%; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | input.default-width { | ||||||
|  |     width: initial; | ||||||
|  | } | ||||||
|  |  | ||||||
| .message-input:placeholder-shown { | .message-input:placeholder-shown { | ||||||
|     border-top: none; |     font-style: italic; | ||||||
|     border-left: none; |  | ||||||
|     border-right: none; |  | ||||||
|     border-bottom-style: dashed; |  | ||||||
|     background-color: #40444b; |     background-color: #40444b; | ||||||
|     color: #fff; |     color: #fff; | ||||||
| } | } | ||||||
| @@ -363,8 +472,7 @@ textarea, input { | |||||||
| .customizable.is-400x300 img { | .customizable.is-400x300 img { | ||||||
|     margin-top: 10px; |     margin-top: 10px; | ||||||
|     width: 100%; |     width: 100%; | ||||||
|     min-height: 100px; |     height: 100px; | ||||||
|     max-height: 400px; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| .customizable.is-32x32 img { | .customizable.is-32x32 img { | ||||||
| @@ -458,6 +566,7 @@ textarea, input { | |||||||
|     flex-grow: 1; |     flex-grow: 1; | ||||||
|     flex-shrink: 1; |     flex-shrink: 1; | ||||||
|     flex-basis: auto; |     flex-basis: auto; | ||||||
|  |     margin-right: 4px; | ||||||
| } | } | ||||||
|  |  | ||||||
| .embed-body input, .embed-body textarea { | .embed-body input, .embed-body textarea { | ||||||
| @@ -507,21 +616,88 @@ textarea, input { | |||||||
|     border-bottom: 1px solid #fff; |     border-bottom: 1px solid #fff; | ||||||
| } | } | ||||||
|  |  | ||||||
| @media only screen and (max-width: 768px) { | .channel-selector { | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select { | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | li.highlight { | ||||||
|  |     margin-bottom: 0 !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .button-row { | ||||||
|  |     display: flex; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .button-row-edit > button { | ||||||
|  |     margin-right: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .button-row .button-row-reminder { | ||||||
|  |     flex-grow: 0; | ||||||
|  |     padding: 2px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .button-row-template { | ||||||
|  |     display: flex; | ||||||
|  |     flex-grow: 1; | ||||||
|  |     justify-content: space-between; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .button-row .button-row-template > div { | ||||||
|  |     padding: 2px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @media only screen and (max-width: 1023px) { | ||||||
|  |     p.title.pageTitle { | ||||||
|  |         display: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .dashboard-frame { | ||||||
|  |         margin-top: 4rem !important; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     .customizable.thumbnail img { |     .customizable.thumbnail img { | ||||||
|         width: 60px; |         width: 60px; | ||||||
|         height: 60px; |         height: 60px; | ||||||
|     } |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|     .customizable.is-24x24 img { | @media only screen and (max-width: 768px) { | ||||||
|         width: 16px; |     .button-row { | ||||||
|         height: 16px; |         display: flex; | ||||||
|  |         flex-direction: column; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .button-row .button-row-reminder { | ||||||
|  |         width: 100%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .button-row .button-row-template > div { | ||||||
|  |         flex-basis: 0; | ||||||
|  |         flex-grow: 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .button-row button { | ||||||
|  |         width: 100%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .reminder-settings { | ||||||
|  |         margin-bottom: 0 !important; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .tts-row { | ||||||
|  |         padding-bottom: 0; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| /* loader */ | /* loader */ | ||||||
| #loader { | #loader { | ||||||
|     position: fixed; |     position: fixed; | ||||||
|  |     top: 0; | ||||||
|     background-color: rgba(255, 255, 255, 0.8); |     background-color: rgba(255, 255, 255, 0.8); | ||||||
|     width: 100vw; |     width: 100vw; | ||||||
|     z-index: 999; |     z-index: 999; | ||||||
| @@ -533,6 +709,86 @@ textarea, input { | |||||||
|  |  | ||||||
| /* END */ | /* END */ | ||||||
|  |  | ||||||
|  | div.reminderError { | ||||||
|  |     margin: 10px; | ||||||
|  |     padding: 14px; | ||||||
|  |     background-color: #f5f5f5; | ||||||
|  |     border-radius: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderError .errorHead { | ||||||
|  |     display: flex; | ||||||
|  |     flex-direction: row; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderError .errorIcon { | ||||||
|  |     padding: 8px; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     margin-right: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderError .errorIcon .fas { | ||||||
|  |     display: none | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderError[data-case="deleted"] .errorIcon { | ||||||
|  |     background-color: #e7e5e4; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderError[data-case="failed"] .errorIcon { | ||||||
|  |     background-color: #fecaca; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderError[data-case="sent"] .errorIcon { | ||||||
|  |     background-color: #d9f99d; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderError[data-case="deleted"] .errorIcon .fas.fa-trash { | ||||||
|  |     display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderError[data-case="failed"] .errorIcon .fas.fa-exclamation-triangle { | ||||||
|  |     display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderError[data-case="sent"] .errorIcon .fas.fa-check { | ||||||
|  |     display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderError .errorHead .reminderName { | ||||||
|  |     font-size: 1rem; | ||||||
|  |     display: flex; | ||||||
|  |     flex-direction: column; | ||||||
|  |     justify-content: center; | ||||||
|  |     color: rgb(54, 54, 54); | ||||||
|  |     flex-grow: 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderError .errorHead .reminderTime { | ||||||
|  |     font-size: 1rem; | ||||||
|  |     display: flex; | ||||||
|  |     flex-direction: column; | ||||||
|  |     flex-shrink: 1; | ||||||
|  |     justify-content: center; | ||||||
|  |     color: rgb(54, 54, 54); | ||||||
|  |     background-color: #ffffff; | ||||||
|  |     padding: 8px; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     border-color: #e5e5e5; | ||||||
|  |     border-width: 1px; | ||||||
|  |     border-style: solid; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.reminderError .reminderMessage { | ||||||
|  |     font-size: 1rem; | ||||||
|  |     display: flex; | ||||||
|  |     flex-direction: column; | ||||||
|  |     justify-content: center; | ||||||
|  |     color: rgb(54, 54, 54); | ||||||
|  |     flex-grow: 1; | ||||||
|  |     font-style: italic; | ||||||
|  | } | ||||||
|  |  | ||||||
| /* other stuff */ | /* other stuff */ | ||||||
|  |  | ||||||
| .half-rem { | .half-rem { | ||||||
| @@ -564,11 +820,44 @@ textarea, input { | |||||||
|     background-color: white; |     background-color: white; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | a.switch-pane { | ||||||
|  |     white-space: nowrap; | ||||||
|  |     overflow: hidden; | ||||||
|  |     text-overflow: ellipsis; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .guild-submenu { | ||||||
|  |     display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .guild-submenu li { | ||||||
|  |     font-size: 0.8rem; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.switch-pane.is-active ~ .guild-submenu { | ||||||
|  |     display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .feedback { | ||||||
|  |     background-color: #5865F2; | ||||||
|  | } | ||||||
|  |  | ||||||
| .is-locked { | .is-locked { | ||||||
|     pointer-events: none; |     pointer-events: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .is-locked > :not(.patreon-invert) { | ||||||
|     opacity: 0.4; |     opacity: 0.4; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .is-locked .patreon-invert { | ||||||
|  |     display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .patreon-invert { | ||||||
|  |     display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
| .is-locked .foreground { | .is-locked .foreground { | ||||||
|     pointer-events: auto; |     pointer-events: auto; | ||||||
| } | } | ||||||
| @@ -576,3 +865,27 @@ textarea, input { | |||||||
| .is-locked .field:last-of-type { | .is-locked .field:last-of-type { | ||||||
|     display: none; |     display: none; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .stat-row { | ||||||
|  |     display: flex; | ||||||
|  |     flex-direction: row; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stat-box { | ||||||
|  |     flex-grow: 1; | ||||||
|  |     border-radius: 6px; | ||||||
|  |     background-color: #fcfcfc; | ||||||
|  |     border-color: #efefef; | ||||||
|  |     border-style: solid; | ||||||
|  |     border-width: 1px; | ||||||
|  |     margin: 4px; | ||||||
|  |     padding: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .figure { | ||||||
|  |     text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .figure-num { | ||||||
|  |     font-size: 2rem; | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								web/static/img/logo_nobg.webp
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 81 KiB | 
							
								
								
									
										
											BIN
										
									
								
								web/static/img/support/delete_reminder/cancel-1.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 17 KiB | 
							
								
								
									
										
											BIN
										
									
								
								web/static/img/support/delete_reminder/cancel-2.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 14 KiB | 
							
								
								
									
										
											BIN
										
									
								
								web/static/img/support/iemanager/edit_spreadsheet.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 44 KiB | 
							
								
								
									
										
											BIN
										
									
								
								web/static/img/support/iemanager/format_text.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 40 KiB | 
							
								
								
									
										
											BIN
										
									
								
								web/static/img/support/iemanager/import.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 44 KiB | 
							
								
								
									
										
											BIN
										
									
								
								web/static/img/support/iemanager/select_export.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 18 KiB | 
							
								
								
									
										
											BIN
										
									
								
								web/static/img/support/iemanager/sheets_settings.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 20 KiB | 
							
								
								
									
										131
									
								
								web/static/js/admin.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,131 @@ | |||||||
|  | document.addEventListener("DOMContentLoaded", () => { | ||||||
|  |     fetch("/admin/data") | ||||||
|  |         .then((resp) => resp.json()) | ||||||
|  |         .then((data) => { | ||||||
|  |             document.querySelector("#backlog").textContent = data.backlog; | ||||||
|  |             document.querySelector("#reminders").textContent = data.count.reminders; | ||||||
|  |             document.querySelector("#intervals").textContent = data.count.intervals; | ||||||
|  |  | ||||||
|  |             let historySent = data.historyLong.sent.reduce( | ||||||
|  |                 (iv, frame) => iv + frame.count, | ||||||
|  |                 0 | ||||||
|  |             ); | ||||||
|  |             let historyFailed = data.historyLong.failed.reduce( | ||||||
|  |                 (iv, frame) => iv + frame.count, | ||||||
|  |                 0 | ||||||
|  |             ); | ||||||
|  |             let rate = historyFailed / (historySent + historyFailed); | ||||||
|  |             let formatted = Math.round(rate * 10000) / 100; | ||||||
|  |  | ||||||
|  |             document.querySelector("#historySent").textContent = historySent; | ||||||
|  |             document.querySelector("#historyFailed").textContent = historyFailed; | ||||||
|  |             document.querySelector("#failRate").textContent = `${formatted}%`; | ||||||
|  |  | ||||||
|  |             new Chart(document.getElementById("schedule"), { | ||||||
|  |                 type: "bar", | ||||||
|  |                 data: { | ||||||
|  |                     labels: [ | ||||||
|  |                         ...data.scheduleShort.once, | ||||||
|  |                         ...data.scheduleShort.interval, | ||||||
|  |                     ].map((row) => luxon.DateTime.fromISO(row.time_key)), | ||||||
|  |                     datasets: [ | ||||||
|  |                         { | ||||||
|  |                             label: "Reminders", | ||||||
|  |                             data: data.scheduleShort.once.map((row) => row.count), | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             label: "Intervals", | ||||||
|  |                             data: data.scheduleShort.interval.map((row) => row.count), | ||||||
|  |                         }, | ||||||
|  |                     ], | ||||||
|  |                 }, | ||||||
|  |                 options: { | ||||||
|  |                     responsive: true, | ||||||
|  |                     maintainAspectRatio: false, | ||||||
|  |                     scales: { | ||||||
|  |                         x: { | ||||||
|  |                             stacked: true, | ||||||
|  |                             type: "time", | ||||||
|  |                             time: { | ||||||
|  |                                 unit: "minute", | ||||||
|  |                             }, | ||||||
|  |                         }, | ||||||
|  |                         y: { | ||||||
|  |                             stacked: true, | ||||||
|  |                         }, | ||||||
|  |                     }, | ||||||
|  |                 }, | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             new Chart(document.getElementById("scheduleLong"), { | ||||||
|  |                 type: "bar", | ||||||
|  |                 data: { | ||||||
|  |                     labels: [ | ||||||
|  |                         ...data.scheduleLong.once, | ||||||
|  |                         ...data.scheduleLong.interval, | ||||||
|  |                     ].map((row) => luxon.DateTime.fromISO(row.time_key)), | ||||||
|  |                     datasets: [ | ||||||
|  |                         { | ||||||
|  |                             label: "Reminders", | ||||||
|  |                             data: data.scheduleLong.once.map((row) => row.count), | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             label: "Intervals", | ||||||
|  |                             data: data.scheduleLong.interval.map((row) => row.count), | ||||||
|  |                         }, | ||||||
|  |                     ], | ||||||
|  |                 }, | ||||||
|  |                 options: { | ||||||
|  |                     responsive: true, | ||||||
|  |                     maintainAspectRatio: false, | ||||||
|  |                     scales: { | ||||||
|  |                         x: { | ||||||
|  |                             stacked: true, | ||||||
|  |                             type: "time", | ||||||
|  |                             time: { | ||||||
|  |                                 unit: "day", | ||||||
|  |                             }, | ||||||
|  |                         }, | ||||||
|  |                         y: { | ||||||
|  |                             stacked: true, | ||||||
|  |                         }, | ||||||
|  |                     }, | ||||||
|  |                 }, | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             new Chart(document.getElementById("historyLong"), { | ||||||
|  |                 type: "bar", | ||||||
|  |                 data: { | ||||||
|  |                     labels: [...data.historyLong.sent, ...data.historyLong.failed].map( | ||||||
|  |                         (row) => luxon.DateTime.fromISO(row.time_key) | ||||||
|  |                     ), | ||||||
|  |                     datasets: [ | ||||||
|  |                         { | ||||||
|  |                             label: "Success", | ||||||
|  |                             data: data.historyLong.sent.map((row) => row.count), | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             label: "Fail", | ||||||
|  |                             data: data.historyLong.failed.map((row) => row.count), | ||||||
|  |                         }, | ||||||
|  |                     ], | ||||||
|  |                 }, | ||||||
|  |                 options: { | ||||||
|  |                     responsive: true, | ||||||
|  |                     maintainAspectRatio: false, | ||||||
|  |                     scales: { | ||||||
|  |                         x: { | ||||||
|  |                             stacked: true, | ||||||
|  |                             type: "time", | ||||||
|  |                             time: { | ||||||
|  |                                 unit: "day", | ||||||
|  |                             }, | ||||||
|  |                         }, | ||||||
|  |                         y: { | ||||||
|  |                             stacked: true, | ||||||
|  |                         }, | ||||||
|  |                     }, | ||||||
|  |                 }, | ||||||
|  |             }); | ||||||
|  |         }); | ||||||
|  | }); | ||||||