Compare commits
	
		
			3 Commits
		
	
	
		
			4416e5d175
			...
			jellywx/gu
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| b0a04bb289 | |||
| eef1f6f3e8 | |||
| 3d08027325 | 
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -2,4 +2,6 @@ | |||||||
| .env | .env | ||||||
| /venv | /venv | ||||||
| .cargo | .cargo | ||||||
|  | assets | ||||||
|  | out.json | ||||||
| /.idea | /.idea | ||||||
|   | |||||||
							
								
								
									
										1213
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1213
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										34
									
								
								Cargo.toml
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								Cargo.toml
									
									
									
									
									
								
							| @@ -1,22 +1,20 @@ | |||||||
| [package] | [package] | ||||||
| name = "reminder-rs" | name = "reminder_rs" | ||||||
| version = "1.6.10" | version = "1.6.6" | ||||||
| authors = ["Jude Southworth <judesouthworth@pm.me>"] | authors = ["jellywx <judesouthworth@pm.me>"] | ||||||
| edition = "2021" | edition = "2018" | ||||||
| license = "AGPL-3.0 only" |  | ||||||
| description = "Reminder Bot for Discord, now in Rust" |  | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| poise = "0.4" | poise = "0.3" | ||||||
| dotenv = "0.15" | dotenv = "0.15" | ||||||
| tokio = { version = "1", features = ["process", "full"] } | tokio = { version = "1", features = ["process", "full"] } | ||||||
| reqwest = "0.11" | reqwest = "0.11" | ||||||
| lazy-regex = "2.3.0" | lazy-regex = "2.3.0" | ||||||
| regex = "1.6" | regex = "1.6" | ||||||
| log = "0.4" | log = "0.4" | ||||||
| env_logger = "0.10" | env_logger = "0.9" | ||||||
| chrono = "0.4" | chrono = "0.4" | ||||||
| chrono-tz = { version = "0.8", features = ["serde"] } | chrono-tz = { version = "0.6", features = ["serde"] } | ||||||
| lazy_static = "1.4" | lazy_static = "1.4" | ||||||
| num-integer = "0.1" | num-integer = "0.1" | ||||||
| serde = "1.0" | serde = "1.0" | ||||||
| @@ -25,7 +23,7 @@ serde_repr = "0.1" | |||||||
| rmp-serde = "1.1" | rmp-serde = "1.1" | ||||||
| rand = "0.8" | rand = "0.8" | ||||||
| levenshtein = "1.0" | levenshtein = "1.0" | ||||||
| sqlx = { version = "0.6", features = ["runtime-tokio-rustls", "macros", "mysql", "bigdecimal", "chrono", "migrate"]} | sqlx = { version = "0.6", features = ["runtime-tokio-rustls", "macros", "mysql", "bigdecimal", "chrono"]} | ||||||
| base64 = "0.13" | base64 = "0.13" | ||||||
|  |  | ||||||
| [dependencies.postman] | [dependencies.postman] | ||||||
| @@ -33,19 +31,3 @@ path = "postman" | |||||||
|  |  | ||||||
| [dependencies.reminder_web] | [dependencies.reminder_web] | ||||||
| path = "web" | path = "web" | ||||||
|  |  | ||||||
| [package.metadata.deb] |  | ||||||
| depends = "$auto, python3-dateparser" |  | ||||||
| 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/default.env", "600"], |  | ||||||
|     ["web/static/**/*", "var/www/reminder-rs/static", "755"], |  | ||||||
|     ["web/templates/**/*", "var/www/reminder-rs/templates", "755"], |  | ||||||
| #    ["nginx/reminder-rs", "etc/nginx/sites-available/reminder-rs", "755"] |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| [package.metadata.deb.systemd-units] |  | ||||||
| unit-scripts = "systemd" |  | ||||||
| start = false |  | ||||||
|   | |||||||
| @@ -1,9 +0,0 @@ | |||||||
| 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 |  | ||||||
							
								
								
									
										42
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										42
									
								
								README.md
									
									
									
									
									
								
							| @@ -7,30 +7,25 @@ reminders are paid on the hosted version of the bot. Keep reading if you want to | |||||||
|  |  | ||||||
| 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 for local target | ### Compiling | ||||||
| 1. Install requirements:  | Install build requirements:  | ||||||
| `sudo apt install gcc gcc-multilib cmake libssl-dev build-essential python3-dateparser` | `sudo apt install gcc gcc-multilib cmake libssl-dev build-essential` | ||||||
| 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` |  | ||||||
|  |  | ||||||
| ### Compiling for other target | Install Rust from https://rustup.rs | ||||||
| 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. |  | ||||||
|  |  | ||||||
| 1. Install container software: `sudo apt install podman`. | Reminder Bot can then be built by running `cargo build --release` in the top level directory. It is necessary to create a  | ||||||
| 2. Install database server: `sudo apt install mysql-server-8.0`. Create a database called `reminders` | folder called 'assets' containing an image file with its name specified in the environment as `WEBHOOK_AVATAR`, of  | ||||||
| 3. Install SQLx CLI: `cargo install sqlx-cli` | dimensions 128x128px to be used as the webhook avatar. | ||||||
| 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`  |  | ||||||
|  |  | ||||||
|  | #### Compilation environment variables | ||||||
|  | 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** | ||||||
|  |  | ||||||
| ### Configuring | ### Setting up Python | ||||||
|  | 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 | ||||||
|  |  | ||||||
|  | ### 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__ | ||||||
| @@ -42,5 +37,10 @@ __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 `/usr/bin/python3`. Can be changed if your Python executable is located somewhere else | * `PYTHON_LOCATION` - default `venv/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
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								Rocket.toml
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| [default] | [default] | ||||||
| address = "0.0.0.0" | address = "0.0.0.0" | ||||||
| port = 18920 | port = 5000 | ||||||
| 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" | ||||||
|  |  | ||||||
| [debug.rsa_sha256.tls] | [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" | ||||||
|  |  | ||||||
| [debug.ecdsa_nistp256_sha256.tls] | [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" | ||||||
|  |  | ||||||
| [debug.ecdsa_nistp384_sha384.tls] | [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" | ||||||
|  |  | ||||||
| [debug.ed25519.tls] | [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" | ||||||
|   | |||||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 21 KiB | 
| @@ -1,16 +0,0 @@ | |||||||
| DATABASE_URL= |  | ||||||
|  |  | ||||||
| DISCORD_TOKEN= |  | ||||||
| PATREON_GUILD_ID= |  | ||||||
| PATREON_ROLE_ID= |  | ||||||
|  |  | ||||||
| LOCAL_TIMEZONE= |  | ||||||
| MIN_INTERVAL= |  | ||||||
| PYTHON_LOCATION=/usr/bin/python3 |  | ||||||
| DONTRUN=web |  | ||||||
| SECRET_KEY= |  | ||||||
|  |  | ||||||
| REMIND_INTERVAL= |  | ||||||
| OAUTH2_DISCORD_CALLBACK= |  | ||||||
| OAUTH2_CLIENT_ID= |  | ||||||
| OAUTH2_CLIENT_SECRET= |  | ||||||
							
								
								
									
										13
									
								
								debian/postinst
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								debian/postinst
									
									
									
									
										vendored
									
									
								
							| @@ -1,13 +0,0 @@ | |||||||
| #!/bin/bash |  | ||||||
|  |  | ||||||
| set -e |  | ||||||
|  |  | ||||||
| id -u reminder &>/dev/null || useradd -r -M reminder |  | ||||||
|  |  | ||||||
| if [ ! -f /etc/reminder-rs/config.env ]; then |  | ||||||
|   cp /etc/reminder-rs/default.env /etc/reminder-rs/config.env |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| chown reminder /etc/reminder-rs/config.env |  | ||||||
|  |  | ||||||
| #DEBHELPER# |  | ||||||
							
								
								
									
										11
									
								
								debian/postrm
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								debian/postrm
									
									
									
									
										vendored
									
									
								
							| @@ -1,11 +0,0 @@ | |||||||
| #!/bin/bash |  | ||||||
|  |  | ||||||
| set -e |  | ||||||
|  |  | ||||||
| id -u reminder &>/dev/null || userdel reminder |  | ||||||
|  |  | ||||||
| if [ -f /etc/reminder-rs/config.env ]; then |  | ||||||
|   rm /etc/reminder-rs/config.env |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| #DEBHELPER# |  | ||||||
| @@ -1,6 +1,10 @@ | |||||||
|  | CREATE DATABASE IF NOT EXISTS reminders; | ||||||
|  | 
 | ||||||
| SET FOREIGN_KEY_CHECKS=0; | SET FOREIGN_KEY_CHECKS=0; | ||||||
| 
 | 
 | ||||||
| CREATE TABLE guilds ( | USE reminders; | ||||||
|  | 
 | ||||||
|  | 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, | ||||||
| 
 | 
 | ||||||
| @@ -14,10 +18,10 @@ CREATE TABLE 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 channels(id) ON DELETE SET NULL |     FOREIGN KEY (default_channel_id) REFERENCES reminders.channels(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE channels ( | CREATE TABLE reminders.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, | ||||||
| 
 | 
 | ||||||
| @@ -35,10 +39,10 @@ CREATE TABLE channels ( | |||||||
|     guild_id INT UNSIGNED, |     guild_id INT UNSIGNED, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE |     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE users ( | CREATE TABLE reminders.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, | ||||||
| 
 | 
 | ||||||
| @@ -55,10 +59,10 @@ CREATE TABLE users ( | |||||||
|     patreon BOOLEAN NOT NULL DEFAULT 0, |     patreon BOOLEAN NOT NULL DEFAULT 0, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (dm_channel) REFERENCES channels(id) ON DELETE RESTRICT |     FOREIGN KEY (dm_channel) REFERENCES reminders.channels(id) ON DELETE RESTRICT | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE roles ( | CREATE TABLE reminders.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, | ||||||
| 
 | 
 | ||||||
| @@ -67,10 +71,10 @@ CREATE TABLE roles ( | |||||||
|     guild_id INT UNSIGNED NOT NULL, |     guild_id INT UNSIGNED NOT NULL, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE |     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE embeds ( | CREATE TABLE reminders.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 '', | ||||||
| @@ -87,7 +91,7 @@ CREATE TABLE embeds ( | |||||||
|     PRIMARY KEY (id) |     PRIMARY KEY (id) | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE embed_fields ( | CREATE TABLE reminders.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 '', | ||||||
| @@ -96,10 +100,10 @@ CREATE TABLE 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 embeds(id) ON DELETE CASCADE |     FOREIGN KEY (embed_id) REFERENCES reminders.embeds(id) ON DELETE CASCADE | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE messages ( | CREATE TABLE reminders.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 '', | ||||||
| @@ -110,10 +114,10 @@ CREATE TABLE messages ( | |||||||
|     attachment_name VARCHAR(260), |     attachment_name VARCHAR(260), | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (embed_id) REFERENCES embeds(id) ON DELETE SET NULL |     FOREIGN KEY (embed_id) REFERENCES reminders.embeds(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE reminders ( | CREATE TABLE reminders.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, | ||||||
| 
 | 
 | ||||||
| @@ -136,20 +140,20 @@ CREATE TABLE reminders ( | |||||||
|     set_by INT UNSIGNED, |     set_by INT UNSIGNED, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (message_id) REFERENCES messages(id) ON DELETE RESTRICT, |     FOREIGN KEY (message_id) REFERENCES reminders.messages(id) ON DELETE RESTRICT, | ||||||
|     FOREIGN KEY (channel_id) REFERENCES channels(id) ON DELETE CASCADE, |     FOREIGN KEY (channel_id) REFERENCES reminders.channels(id) ON DELETE CASCADE, | ||||||
|     FOREIGN KEY (set_by) REFERENCES users(id) ON DELETE SET NULL |     FOREIGN KEY (set_by) REFERENCES reminders.users(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TRIGGER message_cleanup AFTER DELETE ON reminders | CREATE TRIGGER message_cleanup AFTER DELETE ON reminders.reminders | ||||||
| FOR EACH ROW | FOR EACH ROW | ||||||
|     DELETE FROM messages WHERE id = OLD.message_id; |     DELETE FROM reminders.messages WHERE id = OLD.message_id; | ||||||
| 
 | 
 | ||||||
| CREATE TRIGGER embed_cleanup AFTER DELETE ON messages | CREATE TRIGGER embed_cleanup AFTER DELETE ON reminders.messages | ||||||
| FOR EACH ROW | FOR EACH ROW | ||||||
|     DELETE FROM embeds WHERE id = OLD.embed_id; |     DELETE FROM reminders.embeds WHERE id = OLD.embed_id; | ||||||
| 
 | 
 | ||||||
| CREATE TABLE todos ( | CREATE TABLE reminders.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, | ||||||
| @@ -157,23 +161,23 @@ CREATE TABLE todos ( | |||||||
|     value VARCHAR(2000) NOT NULL, |     value VARCHAR(2000) NOT NULL, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE SET NULL, |     FOREIGN KEY (user_id) REFERENCES reminders.users(id) ON DELETE SET NULL, | ||||||
|     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE, |     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE, | ||||||
|     FOREIGN KEY (channel_id) REFERENCES channels(id) ON DELETE SET NULL |     FOREIGN KEY (channel_id) REFERENCES reminders.channels(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE command_restrictions ( | CREATE TABLE reminders.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 roles(id) ON DELETE CASCADE, |     FOREIGN KEY (role_id) REFERENCES reminders.roles(id) ON DELETE CASCADE, | ||||||
|     UNIQUE KEY (`role_id`, `command`) |     UNIQUE KEY (`role_id`, `command`) | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE timers ( | CREATE TABLE reminders.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, | ||||||
| @@ -182,7 +186,7 @@ CREATE TABLE timers ( | |||||||
|     PRIMARY KEY (id) |     PRIMARY KEY (id) | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE events ( | CREATE TABLE reminders.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(), | ||||||
| 
 | 
 | ||||||
| @@ -194,12 +198,12 @@ CREATE TABLE events ( | |||||||
|     reminder_id INT UNSIGNED, |     reminder_id INT UNSIGNED, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE, |     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE, | ||||||
|     FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE SET NULL, |     FOREIGN KEY (user_id) REFERENCES reminders.users(id) ON DELETE SET NULL, | ||||||
|     FOREIGN KEY (reminder_id) REFERENCES reminders(id) ON DELETE SET NULL |     FOREIGN KEY (reminder_id) REFERENCES reminders.reminders(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE command_aliases ( | CREATE TABLE reminders.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, | ||||||
| @@ -208,22 +212,22 @@ CREATE TABLE command_aliases ( | |||||||
|     command VARCHAR(2048) NOT NULL, |     command VARCHAR(2048) NOT NULL, | ||||||
| 
 | 
 | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE, |     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE, | ||||||
|     UNIQUE KEY (`guild_id`, `name`) |     UNIQUE KEY (`guild_id`, `name`) | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE TABLE guild_users ( | CREATE TABLE reminders.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 guilds(id) ON DELETE CASCADE, |     FOREIGN KEY (guild) REFERENCES reminders.guilds(id) ON DELETE CASCADE, | ||||||
|     FOREIGN KEY (user) REFERENCES users(id) ON DELETE CASCADE, |     FOREIGN KEY (user) REFERENCES reminders.users(id) ON DELETE CASCADE, | ||||||
|     UNIQUE KEY (guild, user) |     UNIQUE KEY (guild, user) | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| CREATE EVENT event_cleanup | CREATE EVENT reminders.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 events WHERE `time` < DATE_SUB(NOW(), INTERVAL 5 DAY); | DO DELETE FROM reminders.events WHERE `time` < DATE_SUB(NOW(), INTERVAL 5 DAY); | ||||||
| @@ -1,3 +1,5 @@ | |||||||
|  | USE reminders; | ||||||
|  | 
 | ||||||
| SET FOREIGN_KEY_CHECKS = 0; | SET FOREIGN_KEY_CHECKS = 0; | ||||||
| 
 | 
 | ||||||
| DROP TABLE IF EXISTS reminders_new; | DROP TABLE IF EXISTS reminders_new; | ||||||
| @@ -1,3 +1,5 @@ | |||||||
|  | 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, | ||||||
							
								
								
									
										4
									
								
								migration/03-reminder_variable_intervals.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								migration/03-reminder_variable_intervals.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | 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,3 +1,5 @@ | |||||||
|  | USE reminders; | ||||||
|  | 
 | ||||||
| CREATE TABLE reminder_template ( | CREATE TABLE reminder_template ( | ||||||
|     `id` INT UNSIGNED NOT NULL AUTO_INCREMENT, |     `id` INT UNSIGNED NOT NULL AUTO_INCREMENT, | ||||||
| 
 | 
 | ||||||
							
								
								
									
										92
									
								
								migration/05-restructure-guild-table.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								migration/05-restructure-guild-table.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,92 @@ | |||||||
|  | SET foreign_key_checks = 0; | ||||||
|  |  | ||||||
|  | START TRANSACTION; | ||||||
|  |  | ||||||
|  | -- drop existing constraints | ||||||
|  | ALTER TABLE channels DROP FOREIGN KEY `channels_ibfk_1`; | ||||||
|  | ALTER TABLE command_aliases DROP FOREIGN KEY `command_aliases_ibfk_1`; | ||||||
|  | ALTER TABLE events DROP FOREIGN KEY `events_ibfk_1`; | ||||||
|  | ALTER TABLE guild_users DROP FOREIGN KEY `guild_users_ibfk_1`; | ||||||
|  | ALTER TABLE macro DROP FOREIGN KEY `macro_ibfk_1`; | ||||||
|  | ALTER TABLE roles DROP FOREIGN KEY `roles_ibfk_1`; | ||||||
|  | ALTER TABLE todos DROP FOREIGN KEY `todos_ibfk_2`; | ||||||
|  | ALTER TABLE reminder_template DROP FOREIGN KEY `reminder_template_ibfk_1`; | ||||||
|  |  | ||||||
|  | -- update foreign key types | ||||||
|  | ALTER TABLE channels MODIFY `guild_id` BIGINT UNSIGNED; | ||||||
|  | ALTER TABLE command_aliases MODIFY `guild_id` BIGINT UNSIGNED; | ||||||
|  | ALTER TABLE events MODIFY `guild_id` BIGINT UNSIGNED; | ||||||
|  | ALTER TABLE guild_users MODIFY `guild` BIGINT UNSIGNED; | ||||||
|  | ALTER TABLE macro MODIFY `guild_id` BIGINT UNSIGNED; | ||||||
|  | ALTER TABLE roles MODIFY `guild_id` BIGINT UNSIGNED; | ||||||
|  | ALTER TABLE todos MODIFY `guild_id` BIGINT UNSIGNED; | ||||||
|  | ALTER TABLE reminder_template MODIFY `guild_id` BIGINT UNSIGNED; | ||||||
|  |  | ||||||
|  | -- update foreign key values | ||||||
|  | UPDATE channels SET `guild_id` = (SELECT `guild` FROM guilds WHERE guilds.`id` = `guild_id`); | ||||||
|  | UPDATE command_aliases SET `guild_id` = (SELECT `guild` FROM guilds WHERE guilds.`id` = `guild_id`); | ||||||
|  | UPDATE events SET `guild_id` = (SELECT `guild` FROM guilds WHERE guilds.`id` = `guild_id`); | ||||||
|  | UPDATE guild_users SET `guild` = (SELECT `guild` FROM guilds WHERE guilds.`id` = guild_users.`guild`); | ||||||
|  | UPDATE macro SET `guild_id` = (SELECT `guild` FROM guilds WHERE guilds.`id` = `guild_id`); | ||||||
|  | UPDATE roles SET `guild_id` = (SELECT `guild` FROM guilds WHERE guilds.`id` = `guild_id`); | ||||||
|  | UPDATE todos SET `guild_id` = (SELECT `guild` FROM guilds WHERE guilds.`id` = `guild_id`); | ||||||
|  | UPDATE reminder_template SET `guild_id` = (SELECT `guild` FROM guilds WHERE guilds.`id` = `guild_id`); | ||||||
|  |  | ||||||
|  | -- update guilds table | ||||||
|  | ALTER TABLE guilds MODIFY `id` BIGINT UNSIGNED NOT NULL; | ||||||
|  | UPDATE guilds SET `id` = `guild`; | ||||||
|  | ALTER TABLE guilds DROP COLUMN `guild`; | ||||||
|  | ALTER TABLE guilds ADD COLUMN `default_channel` BIGINT UNSIGNED; | ||||||
|  | ALTER TABLE guilds ADD CONSTRAINT `default_channel_fk` | ||||||
|  |     FOREIGN KEY (`default_channel`) | ||||||
|  |         REFERENCES channels(`channel`) | ||||||
|  |         ON DELETE SET NULL | ||||||
|  |         ON UPDATE CASCADE; | ||||||
|  |  | ||||||
|  | -- re-add constraints | ||||||
|  | ALTER TABLE channels ADD CONSTRAINT | ||||||
|  |     FOREIGN KEY (`guild_id`) | ||||||
|  |         REFERENCES guilds(`id`) | ||||||
|  |         ON DELETE CASCADE | ||||||
|  |         ON UPDATE CASCADE; | ||||||
|  |  | ||||||
|  | ALTER TABLE command_aliases ADD CONSTRAINT | ||||||
|  |     FOREIGN KEY (`guild_id`) | ||||||
|  |         REFERENCES guilds(`id`) | ||||||
|  |         ON DELETE CASCADE | ||||||
|  |         ON UPDATE CASCADE; | ||||||
|  |  | ||||||
|  | ALTER TABLE events ADD CONSTRAINT | ||||||
|  |     FOREIGN KEY (`guild_id`) | ||||||
|  |         REFERENCES guilds(`id`) | ||||||
|  |         ON DELETE CASCADE | ||||||
|  |         ON UPDATE CASCADE; | ||||||
|  |  | ||||||
|  | ALTER TABLE guild_users ADD CONSTRAINT | ||||||
|  |     FOREIGN KEY (`guild`) | ||||||
|  |         REFERENCES guilds(`id`) | ||||||
|  |         ON DELETE CASCADE | ||||||
|  |         ON UPDATE CASCADE; | ||||||
|  |  | ||||||
|  | ALTER TABLE macro ADD CONSTRAINT | ||||||
|  |     FOREIGN KEY (`guild_id`) | ||||||
|  |         REFERENCES guilds(`id`) | ||||||
|  |         ON DELETE CASCADE | ||||||
|  |         ON UPDATE CASCADE; | ||||||
|  |  | ||||||
|  | ALTER TABLE roles ADD CONSTRAINT | ||||||
|  |     FOREIGN KEY (`guild_id`) | ||||||
|  |         REFERENCES guilds(`id`) | ||||||
|  |         ON DELETE CASCADE | ||||||
|  |         ON UPDATE CASCADE; | ||||||
|  |  | ||||||
|  | ALTER TABLE todos ADD CONSTRAINT | ||||||
|  |     FOREIGN KEY (`guild_id`) | ||||||
|  |         REFERENCES guilds(`id`) | ||||||
|  |         ON DELETE CASCADE | ||||||
|  |         ON UPDATE CASCADE; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | COMMIT; | ||||||
|  |  | ||||||
|  | SET foreign_key_checks = 1; | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| ALTER TABLE reminders RENAME COLUMN `interval` TO `interval_seconds`; |  | ||||||
| ALTER TABLE reminders ADD COLUMN `interval_months` INT UNSIGNED DEFAULT NULL; |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| ALTER TABLE reminders ADD COLUMN `interval_days` INT UNSIGNED DEFAULT NULL; |  | ||||||
| @@ -1,41 +0,0 @@ | |||||||
| 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; |  | ||||||
|         } |  | ||||||
| } |  | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| use chrono::{DateTime, Days, Duration, Months}; | use chrono::Duration; | ||||||
| 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}; | ||||||
| @@ -62,8 +62,7 @@ pub fn substitute(string: &str) -> String { | |||||||
|         let format = caps.name("format").map(|m| m.as_str()); |         let format = caps.name("format").map(|m| m.as_str()); | ||||||
|  |  | ||||||
|         if let (Some(final_time), Some(format)) = (final_time, format) { |         if let (Some(final_time), Some(format)) = (final_time, format) { | ||||||
|             match NaiveDateTime::from_timestamp_opt(final_time, 0) { |             let dt = NaiveDateTime::from_timestamp(final_time, 0); | ||||||
|                 Some(dt) => { |  | ||||||
|             let now = Utc::now().naive_utc(); |             let now = Utc::now().naive_utc(); | ||||||
|  |  | ||||||
|             let difference = { |             let difference = { | ||||||
| @@ -75,10 +74,6 @@ pub fn substitute(string: &str) -> String { | |||||||
|             }; |             }; | ||||||
|  |  | ||||||
|             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() | ||||||
|         } |         } | ||||||
| @@ -248,12 +243,11 @@ pub struct Reminder { | |||||||
|     attachment: Option<Vec<u8>>, |     attachment: Option<Vec<u8>>, | ||||||
|     attachment_name: Option<String>, |     attachment_name: Option<String>, | ||||||
|  |  | ||||||
|     utc_time: DateTime<Utc>, |     utc_time: NaiveDateTime, | ||||||
|     timezone: String, |     timezone: String, | ||||||
|     restartable: bool, |     restartable: bool, | ||||||
|     expires: Option<DateTime<Utc>>, |     expires: Option<NaiveDateTime>, | ||||||
|     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>, | ||||||
| @@ -287,7 +281,6 @@ SELECT | |||||||
|     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, | ||||||
| @@ -337,7 +330,9 @@ WHERE | |||||||
|  |  | ||||||
|     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) | ||||||
| @@ -346,43 +341,55 @@ WHERE | |||||||
|  |  | ||||||
|     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() || self.interval_months.is_some() { | ||||||
|             let now = Utc::now(); |             let now = Utc::now().naive_local(); | ||||||
|             let mut updated_reminder_time = |             let mut updated_reminder_time = self.utc_time; | ||||||
|                 self.utc_time.with_timezone(&self.timezone.parse().unwrap_or(Tz::UTC)); |  | ||||||
|  |  | ||||||
|             while updated_reminder_time < now { |  | ||||||
|             if let Some(interval) = self.interval_months { |             if let Some(interval) = self.interval_months { | ||||||
|                     updated_reminder_time = updated_reminder_time |                 match sqlx::query!( | ||||||
|                         .checked_add_months(Months::new(interval)) |                     // use the second date_add to force return value to datetime | ||||||
|                         .unwrap_or_else(|| { |                     "SELECT DATE_ADD(DATE_ADD(?, INTERVAL ? MONTH), INTERVAL 0 SECOND) AS new_time", | ||||||
|                             warn!("Could not add months to a reminder"); |                     updated_reminder_time, | ||||||
|  |                     interval | ||||||
|                             updated_reminder_time |                 ) | ||||||
|                         }); |                 .fetch_one(pool) | ||||||
|  |                 .await | ||||||
|  |                 { | ||||||
|  |                     Ok(row) => match row.new_time { | ||||||
|  |                         Some(datetime) => { | ||||||
|  |                             updated_reminder_time = datetime; | ||||||
|                         } |                         } | ||||||
|  |                         None => { | ||||||
|  |                             warn!("Could not update interval by months: got NULL"); | ||||||
|  |  | ||||||
|                 if let Some(interval) = self.interval_days { |                             updated_reminder_time += Duration::days(30); | ||||||
|                     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"); |  | ||||||
|  |  | ||||||
|                             updated_reminder_time |                     Err(e) => { | ||||||
|                         }); |                         warn!("Could not update interval by months: {:?}", e); | ||||||
|  |  | ||||||
|  |                         // naively fallback to adding 30 days | ||||||
|  |                         updated_reminder_time += Duration::days(30); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             if let Some(interval) = self.interval_seconds { |             if let Some(interval) = self.interval_seconds { | ||||||
|                     updated_reminder_time = |                 while updated_reminder_time < now { | ||||||
|                         updated_reminder_time + Duration::seconds(interval as i64); |                     updated_reminder_time += Duration::seconds(interval as i64); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             if self.expires.map_or(false, |expires| updated_reminder_time > expires) { |             if self.expires.map_or(false, |expires| { | ||||||
|  |                 NaiveDateTime::from_timestamp(updated_reminder_time.timestamp(), 0) > expires | ||||||
|  |             }) { | ||||||
|                 self.force_delete(pool).await; |                 self.force_delete(pool).await; | ||||||
|             } else { |             } else { | ||||||
|                 sqlx::query!( |                 sqlx::query!( | ||||||
|                     "UPDATE reminders SET `utc_time` = ? WHERE `id` = ?", |                     " | ||||||
|                     updated_reminder_time.with_timezone(&Utc), | UPDATE reminders SET `utc_time` = ? WHERE `id` = ? | ||||||
|  |                     ", | ||||||
|  |                     updated_reminder_time, | ||||||
|                     self.id |                     self.id | ||||||
|                 ) |                 ) | ||||||
|                 .execute(pool) |                 .execute(pool) | ||||||
| @@ -395,7 +402,12 @@ WHERE | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     async fn force_delete(&self, pool: impl Executor<'_, Database = Database> + Copy) { |     async fn force_delete(&self, pool: impl Executor<'_, Database = Database> + Copy) { | ||||||
|         sqlx::query!("DELETE FROM reminders WHERE `id` = ?", self.id) |         sqlx::query!( | ||||||
|  |             " | ||||||
|  | DELETE FROM reminders WHERE `id` = ? | ||||||
|  |             ", | ||||||
|  |             self.id | ||||||
|  |         ) | ||||||
|         .execute(pool) |         .execute(pool) | ||||||
|         .await |         .await | ||||||
|         .expect(&format!("Could not delete Reminder {}", self.id)); |         .expect(&format!("Could not delete Reminder {}", self.id)); | ||||||
| @@ -492,10 +504,8 @@ WHERE | |||||||
|                     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 { | ||||||
|                         if !username.is_empty() { |  | ||||||
|                         w.username(username); |                         w.username(username); | ||||||
|                     } |                     } | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     if let Some(avatar) = &reminder.avatar { |                     if let Some(avatar) = &reminder.avatar { | ||||||
|                         w.avatar_url(avatar); |                         w.avatar_url(avatar); | ||||||
| @@ -538,7 +548,9 @@ WHERE | |||||||
|                     .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) | ||||||
|   | |||||||
| @@ -24,7 +24,7 @@ pub async fn macro_name_autocomplete(ctx: Context<'_>, partial: &str) -> Vec<Str | |||||||
| SELECT name | SELECT name | ||||||
| FROM macro | FROM macro | ||||||
| WHERE | WHERE | ||||||
|     guild_id = (SELECT id FROM guilds WHERE guild = ?) |     guild_id = ? | ||||||
|     AND name LIKE CONCAT(?, '%')", |     AND name LIKE CONCAT(?, '%')", | ||||||
|         ctx.guild_id().unwrap().0, |         ctx.guild_id().unwrap().0, | ||||||
|         partial, |         partial, | ||||||
| @@ -37,6 +37,20 @@ WHERE | |||||||
|     .collect() |     .collect() | ||||||
| } | } | ||||||
|  |  | ||||||
|  | pub async fn multiline_autocomplete( | ||||||
|  |     _ctx: Context<'_>, | ||||||
|  |     partial: &str, | ||||||
|  | ) -> Vec<AutocompleteChoice<String>> { | ||||||
|  |     if partial.is_empty() { | ||||||
|  |         vec![AutocompleteChoice { name: "Multiline content...".to_string(), value: "".to_string() }] | ||||||
|  |     } else { | ||||||
|  |         vec![ | ||||||
|  |             AutocompleteChoice { name: partial.to_string(), value: partial.to_string() }, | ||||||
|  |             AutocompleteChoice { name: "Multiline content...".to_string(), value: "".to_string() }, | ||||||
|  |         ] | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| pub async fn time_hint_autocomplete( | pub async fn time_hint_autocomplete( | ||||||
|     ctx: Context<'_>, |     ctx: Context<'_>, | ||||||
|     partial: &str, |     partial: &str, | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ pub async fn delete_macro( | |||||||
| ) -> Result<(), Error> { | ) -> Result<(), Error> { | ||||||
|     match sqlx::query!( |     match sqlx::query!( | ||||||
|         " |         " | ||||||
| SELECT id FROM macro WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?) AND name = ?", | SELECT id FROM macro WHERE guild_id = ? AND name = ?", | ||||||
|         ctx.guild_id().unwrap().0, |         ctx.guild_id().unwrap().0, | ||||||
|         name |         name | ||||||
|     ) |     ) | ||||||
|   | |||||||
| @@ -24,7 +24,7 @@ pub async fn migrate_macro(ctx: Context<'_>) -> Result<(), Error> { | |||||||
|  |  | ||||||
|     let aliases = sqlx::query_as!( |     let aliases = sqlx::query_as!( | ||||||
|         Alias, |         Alias, | ||||||
|         "SELECT name, command FROM command_aliases WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?)", |         "SELECT name, command FROM command_aliases WHERE guild_id = ?", | ||||||
|         guild_id.0 |         guild_id.0 | ||||||
|     ) |     ) | ||||||
|     .fetch_all(&mut transaction) |     .fetch_all(&mut transaction) | ||||||
| @@ -36,7 +36,7 @@ pub async fn migrate_macro(ctx: Context<'_>) -> Result<(), Error> { | |||||||
|         match parse_text_command(guild_id, alias.name, &alias.command) { |         match parse_text_command(guild_id, alias.name, &alias.command) { | ||||||
|             Some(cmd_macro) => { |             Some(cmd_macro) => { | ||||||
|                 sqlx::query!( |                 sqlx::query!( | ||||||
|                     "INSERT INTO macro (guild_id, name, description, commands) VALUES ((SELECT id FROM guilds WHERE guild = ?), ?, ?, ?)", |                     "INSERT INTO macro (guild_id, name, description, commands) VALUES (?, ?, ?, ?)", | ||||||
|                     cmd_macro.guild_id.0, |                     cmd_macro.guild_id.0, | ||||||
|                     cmd_macro.name, |                     cmd_macro.name, | ||||||
|                     cmd_macro.description, |                     cmd_macro.description, | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ pub async fn record_macro( | |||||||
|  |  | ||||||
|     let row = sqlx::query!( |     let row = sqlx::query!( | ||||||
|         " |         " | ||||||
| SELECT 1 as _e FROM macro WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?) AND name = ?", | SELECT 1 as _e FROM macro WHERE guild_id = ? AND name = ?", | ||||||
|         guild_id.0, |         guild_id.0, | ||||||
|         name |         name | ||||||
|     ) |     ) | ||||||
| @@ -121,7 +121,7 @@ pub async fn finish_macro(ctx: Context<'_>) -> Result<(), Error> { | |||||||
|             let json = serde_json::to_string(&command_macro.commands).unwrap(); |             let json = serde_json::to_string(&command_macro.commands).unwrap(); | ||||||
|  |  | ||||||
|             sqlx::query!( |             sqlx::query!( | ||||||
|                 "INSERT INTO macro (guild_id, name, description, commands) VALUES ((SELECT id FROM guilds WHERE guild = ?), ?, ?, ?)", |                 "INSERT INTO macro (guild_id, name, description, commands) VALUES (?, ?, ?, ?)", | ||||||
|                 command_macro.guild_id.0, |                 command_macro.guild_id.0, | ||||||
|                 command_macro.name, |                 command_macro.name, | ||||||
|                 command_macro.description, |                 command_macro.description, | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| use super::super::autocomplete::macro_name_autocomplete; | use super::super::autocomplete::macro_name_autocomplete; | ||||||
| use crate::{models::command_macro::guild_command_macro, Context, Data, Error, THEME_COLOR}; | use crate::{models::command_macro::guild_command_macro, Context, Data, Error}; | ||||||
|  |  | ||||||
| /// Run a recorded macro | /// Run a recorded macro | ||||||
| #[poise::command( | #[poise::command( | ||||||
| @@ -17,17 +17,7 @@ pub async fn run_macro( | |||||||
| ) -> Result<(), Error> { | ) -> Result<(), Error> { | ||||||
|     match guild_command_macro(&Context::Application(ctx), &name).await { |     match guild_command_macro(&Context::Application(ctx), &name).await { | ||||||
|         Some(command_macro) => { |         Some(command_macro) => { | ||||||
|             Context::Application(ctx) |             ctx.defer_response(false).await?; | ||||||
|                 .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 { |             for command in command_macro.commands { | ||||||
|                 if let Some(action) = command.action { |                 if let Some(action) = command.action { | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ use chrono::offset::Utc; | |||||||
| use chrono_tz::{Tz, TZ_VARIANTS}; | use chrono_tz::{Tz, TZ_VARIANTS}; | ||||||
| use levenshtein::levenshtein; | use levenshtein::levenshtein; | ||||||
| use log::warn; | use log::warn; | ||||||
|  | use poise::serenity_prelude::{ChannelId, Mentionable}; | ||||||
|  |  | ||||||
| use super::autocomplete::timezone_autocomplete; | use super::autocomplete::timezone_autocomplete; | ||||||
| use crate::{consts::THEME_COLOR, models::CtxData, Context, Error}; | use crate::{consts::THEME_COLOR, models::CtxData, Context, Error}; | ||||||
| @@ -148,11 +149,51 @@ pub async fn unset_allowed_dm(ctx: Context<'_>) -> Result<(), Error> { | |||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// Set defaults for commands | ||||||
|  | #[poise::command( | ||||||
|  |     slash_command, | ||||||
|  |     identifying_name = "default", | ||||||
|  |     default_member_permissions = "MANAGE_GUILD" | ||||||
|  | )] | ||||||
|  | pub async fn default(_ctx: Context<'_>) -> Result<(), Error> { | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /// Set a default channel for reminders to be sent to | ||||||
|  | #[poise::command( | ||||||
|  |     slash_command, | ||||||
|  |     guild_only = true, | ||||||
|  |     identifying_name = "default_channel", | ||||||
|  |     default_member_permissions = "MANAGE_GUILD" | ||||||
|  | )] | ||||||
|  | pub async fn default_channel( | ||||||
|  |     ctx: Context<'_>, | ||||||
|  |     #[description = "Channel to send reminders to by default"] channel: Option<ChannelId>, | ||||||
|  | ) -> Result<(), Error> { | ||||||
|  |     if let Some(mut guild_data) = ctx.guild_data().await { | ||||||
|  |         guild_data.default_channel = channel.map(|c| c.0); | ||||||
|  |  | ||||||
|  |         guild_data.commit_changes(&ctx.data().database).await?; | ||||||
|  |  | ||||||
|  |         if let Some(channel) = channel { | ||||||
|  |             ctx.send(|r| { | ||||||
|  |                 r.ephemeral(true).content(format!("Default channel set to {}", channel.mention())) | ||||||
|  |             }) | ||||||
|  |             .await?; | ||||||
|  |         } else { | ||||||
|  |             ctx.send(|r| r.ephemeral(true).content("Default channel unset.")).await?; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  |  | ||||||
| /// View the webhook being used to send reminders to this channel | /// View the webhook being used to send reminders to this channel | ||||||
| #[poise::command( | #[poise::command( | ||||||
|     slash_command, |     slash_command, | ||||||
|     identifying_name = "webhook_url", |     identifying_name = "webhook_url", | ||||||
|     required_permissions = "ADMINISTRATOR" |     required_permissions = "ADMINISTRATOR", | ||||||
|  |     default_member_permissions = "ADMINISTRATOR" | ||||||
| )] | )] | ||||||
| pub async fn webhook(ctx: Context<'_>) -> Result<(), Error> { | pub async fn webhook(ctx: Context<'_>) -> Result<(), Error> { | ||||||
|     match ctx.channel_data().await { |     match ctx.channel_data().await { | ||||||
|   | |||||||
| @@ -1,6 +1,10 @@ | |||||||
| use std::{collections::HashSet, string::ToString}; | use std::{ | ||||||
|  |     collections::HashSet, | ||||||
|  |     string::ToString, | ||||||
|  |     time::{SystemTime, UNIX_EPOCH}, | ||||||
|  | }; | ||||||
|  |  | ||||||
| use chrono::{DateTime, NaiveDateTime, Utc}; | use chrono::NaiveDateTime; | ||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
| use num_integer::Integer; | use num_integer::Integer; | ||||||
| use poise::{ | use poise::{ | ||||||
| @@ -11,7 +15,9 @@ use poise::{ | |||||||
| }; | }; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
|     commands::autocomplete::{time_hint_autocomplete, timezone_autocomplete}, |     commands::autocomplete::{ | ||||||
|  |         multiline_autocomplete, time_hint_autocomplete, timezone_autocomplete, | ||||||
|  |     }, | ||||||
|     component_models::{ |     component_models::{ | ||||||
|         pager::{DelPager, LookPager, Pager}, |         pager::{DelPager, LookPager, Pager}, | ||||||
|         ComponentDataModel, DelSelector, UndoReminder, |         ComponentDataModel, DelSelector, UndoReminder, | ||||||
| @@ -56,8 +62,8 @@ 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 { | ||||||
|                 match NaiveDateTime::from_timestamp_opt(timestamp, 0) { |                 let dt = NaiveDateTime::from_timestamp(timestamp, 0); | ||||||
|                     Some(dt) => { |  | ||||||
|                 channel.paused = true; |                 channel.paused = true; | ||||||
|                 channel.paused_until = Some(dt); |                 channel.paused_until = Some(dt); | ||||||
|  |  | ||||||
| @@ -68,15 +74,6 @@ pub async fn pause( | |||||||
|                     timestamp |                     timestamp | ||||||
|                 )) |                 )) | ||||||
|                 .await?; |                 .await?; | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     None => { |  | ||||||
|                         ctx.say(format!( |  | ||||||
|                             "Time processed could not be interpreted as `DateTime`. Please write the time as clearly as possible", |  | ||||||
|                         )) |  | ||||||
|                         .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", | ||||||
| @@ -250,7 +247,7 @@ pub async fn look( | |||||||
|                 char_count < EMBED_DESCRIPTION_MAX_LENGTH |                 char_count < EMBED_DESCRIPTION_MAX_LENGTH | ||||||
|             }) |             }) | ||||||
|             .collect::<Vec<String>>() |             .collect::<Vec<String>>() | ||||||
|             .join(""); |             .join("\n"); | ||||||
|  |  | ||||||
|         let pages = reminders |         let pages = reminders | ||||||
|             .iter() |             .iter() | ||||||
| @@ -437,8 +434,11 @@ pub fn show_delete_page(reminders: &[Reminder], page: usize, timezone: Tz) -> Cr | |||||||
|     reply |     reply | ||||||
| } | } | ||||||
|  |  | ||||||
| fn time_difference(start_time: DateTime<Utc>) -> String { | fn time_difference(start_time: NaiveDateTime) -> String { | ||||||
|     let delta = (Utc::now() - start_time).num_seconds(); |     let unix_time = SystemTime::now().duration_since(UNIX_EPOCH).unwrap().as_secs() as i64; | ||||||
|  |     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); | ||||||
| @@ -562,17 +562,20 @@ struct ContentModal { | |||||||
|     content: String, |     content: String, | ||||||
| } | } | ||||||
|  |  | ||||||
| /// Create a reminder with multi-line content. Press "+4 more" for other options. | /// Create a reminder. Press "+4 more" for other options. | ||||||
| #[poise::command( | #[poise::command( | ||||||
|     slash_command, |     slash_command, | ||||||
|     identifying_name = "multiline", |     identifying_name = "remind", | ||||||
|     default_member_permissions = "MANAGE_GUILD" |     default_member_permissions = "MANAGE_GUILD" | ||||||
| )] | )] | ||||||
| pub async fn multiline( | pub async fn remind( | ||||||
|     ctx: ApplicationContext<'_>, |     ctx: ApplicationContext<'_>, | ||||||
|     #[description = "A description of the time to set the reminder for"] |     #[description = "A description of the time to set the reminder for"] | ||||||
|     #[autocomplete = "time_hint_autocomplete"] |     #[autocomplete = "time_hint_autocomplete"] | ||||||
|     time: String, |     time: String, | ||||||
|  |     #[description = "The message content to send"] | ||||||
|  |     #[autocomplete = "multiline_autocomplete"] | ||||||
|  |     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>, | ||||||
| @@ -585,6 +588,8 @@ pub async fn multiline( | |||||||
|     timezone: Option<String>, |     timezone: Option<String>, | ||||||
| ) -> Result<(), Error> { | ) -> Result<(), Error> { | ||||||
|     let tz = timezone.map(|t| t.parse::<Tz>().ok()).flatten(); |     let tz = timezone.map(|t| t.parse::<Tz>().ok()).flatten(); | ||||||
|  |  | ||||||
|  |     if content.is_empty() { | ||||||
|         let data = ContentModal::execute(ctx).await?; |         let data = ContentModal::execute(ctx).await?; | ||||||
|  |  | ||||||
|         create_reminder( |         create_reminder( | ||||||
| @@ -598,35 +603,19 @@ pub async fn multiline( | |||||||
|             tz, |             tz, | ||||||
|         ) |         ) | ||||||
|         .await |         .await | ||||||
| } |     } else { | ||||||
|  |         create_reminder( | ||||||
| /// Create a reminder. Press "+4 more" for other options. Use "/multiline" for multiline content. |             Context::Application(ctx), | ||||||
| #[poise::command( |             time, | ||||||
|     slash_command, |             content, | ||||||
|     identifying_name = "remind", |             channels, | ||||||
|     default_member_permissions = "MANAGE_GUILD" |             interval, | ||||||
| )] |             expires, | ||||||
| pub async fn remind( |             tts, | ||||||
|     ctx: ApplicationContext<'_>, |             tz, | ||||||
|     #[description = "A description of the time to set the reminder for"] |         ) | ||||||
|     #[autocomplete = "time_hint_autocomplete"] |  | ||||||
|     time: String, |  | ||||||
|     #[description = "The message content to send"] content: 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(); |  | ||||||
|  |  | ||||||
|     create_reminder(Context::Application(ctx), time, content, channels, interval, expires, tts, tz) |  | ||||||
|         .await |         .await | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| async fn create_reminder( | async fn create_reminder( | ||||||
| @@ -664,7 +653,9 @@ async fn create_reminder( | |||||||
|                 let list = channels.map(|arg| parse_mention_list(&arg)).unwrap_or_default(); |                 let list = channels.map(|arg| parse_mention_list(&arg)).unwrap_or_default(); | ||||||
|  |  | ||||||
|                 if list.is_empty() { |                 if list.is_empty() { | ||||||
|                     if ctx.guild_id().is_some() { |                     if let Some(channel_id) = ctx.default_channel().await { | ||||||
|  |                         vec![ReminderScope::Channel(channel_id.0)] | ||||||
|  |                     } else if ctx.guild_id().is_some() { | ||||||
|                         vec![ReminderScope::Channel(ctx.channel_id().0)] |                         vec![ReminderScope::Channel(ctx.channel_id().0)] | ||||||
|                     } else { |                     } else { | ||||||
|                         vec![ReminderScope::User(ctx.author().id.0)] |                         vec![ReminderScope::User(ctx.author().id.0)] | ||||||
|   | |||||||
| @@ -47,7 +47,7 @@ pub async fn todo_guild_add( | |||||||
| ) -> Result<(), Error> { | ) -> Result<(), Error> { | ||||||
|     sqlx::query!( |     sqlx::query!( | ||||||
|         "INSERT INTO todos (guild_id, value) |         "INSERT INTO todos (guild_id, value) | ||||||
| VALUES ((SELECT id FROM guilds WHERE guild = ?), ?)", | VALUES (?, ?)", | ||||||
|         ctx.guild_id().unwrap().0, |         ctx.guild_id().unwrap().0, | ||||||
|         task |         task | ||||||
|     ) |     ) | ||||||
| @@ -70,9 +70,7 @@ VALUES ((SELECT id FROM guilds WHERE guild = ?), ?)", | |||||||
| )] | )] | ||||||
| pub async fn todo_guild_view(ctx: Context<'_>) -> Result<(), Error> { | pub async fn todo_guild_view(ctx: Context<'_>) -> Result<(), Error> { | ||||||
|     let values = sqlx::query!( |     let values = sqlx::query!( | ||||||
|         "SELECT todos.id, value FROM todos |         "SELECT todos.id, value FROM todos WHERE guild_id = ?", | ||||||
| INNER JOIN guilds ON todos.guild_id = guilds.id |  | ||||||
| WHERE guilds.guild = ?", |  | ||||||
|         ctx.guild_id().unwrap().0, |         ctx.guild_id().unwrap().0, | ||||||
|     ) |     ) | ||||||
|     .fetch_all(&ctx.data().database) |     .fetch_all(&ctx.data().database) | ||||||
| @@ -122,7 +120,7 @@ pub async fn todo_channel_add( | |||||||
|  |  | ||||||
|     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 channels WHERE channel = ?), ?)", | ||||||
|         ctx.guild_id().unwrap().0, |         ctx.guild_id().unwrap().0, | ||||||
|         ctx.channel_id().0, |         ctx.channel_id().0, | ||||||
|         task |         task | ||||||
| @@ -340,18 +338,7 @@ 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({ |                                         .description(disp.split_once(' ').unwrap_or(("", "")).1) | ||||||
|                                             let c = disp.split_once(' ').unwrap_or(("", "")).1; |  | ||||||
|  |  | ||||||
|                                             if c.len() > 100 { |  | ||||||
|                                                 format!( |  | ||||||
|                                                     "{}...", |  | ||||||
|                                                     c.chars().take(97).collect::<String>() |  | ||||||
|                                                 ) |  | ||||||
|                                             } else { |  | ||||||
|                                                 c.to_string() |  | ||||||
|                                             } |  | ||||||
|                                         }) |  | ||||||
|                                 }); |                                 }); | ||||||
|                             } |                             } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -113,7 +113,7 @@ impl ComponentDataModel { | |||||||
|                         char_count < EMBED_DESCRIPTION_MAX_LENGTH |                         char_count < EMBED_DESCRIPTION_MAX_LENGTH | ||||||
|                     }) |                     }) | ||||||
|                     .collect::<Vec<String>>() |                     .collect::<Vec<String>>() | ||||||
|                     .join(""); |                     .join("\n"); | ||||||
|  |  | ||||||
|                 let mut embed = CreateEmbed::default(); |                 let mut embed = CreateEmbed::default(); | ||||||
|                 embed |                 embed | ||||||
| @@ -222,9 +222,7 @@ WHERE channels.channel = ?", | |||||||
|                         .collect::<Vec<(usize, String)>>() |                         .collect::<Vec<(usize, String)>>() | ||||||
|                     } else { |                     } else { | ||||||
|                         sqlx::query!( |                         sqlx::query!( | ||||||
|                             "SELECT todos.id, value FROM todos |                             "SELECT todos.id, value FROM todos WHERE guild_id = ?", | ||||||
| INNER JOIN guilds ON todos.guild_id = guilds.id |  | ||||||
| WHERE guilds.guild = ?", |  | ||||||
|                             pager.guild_id, |                             pager.guild_id, | ||||||
|                         ) |                         ) | ||||||
|                         .fetch_all(&data.database) |                         .fetch_all(&data.database) | ||||||
|   | |||||||
| @@ -17,13 +17,17 @@ use regex::Regex; | |||||||
|  |  | ||||||
| lazy_static! { | lazy_static! { | ||||||
|     pub static ref DEFAULT_AVATAR: AttachmentType<'static> = ( |     pub static ref DEFAULT_AVATAR: AttachmentType<'static> = ( | ||||||
|         include_bytes!(concat!(env!("CARGO_MANIFEST_DIR"), "/assets/webhook.jpg")) as &[u8], |         include_bytes!(concat!( | ||||||
|         "webhook.jpg", |             env!("CARGO_MANIFEST_DIR"), | ||||||
|  |             "/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("PATREON_ROLE_ID") |         env::var("SUBSCRIPTION_ROLES") | ||||||
|             .map(|var| var |             .map(|var| var | ||||||
|                 .split(',') |                 .split(',') | ||||||
|                 .filter_map(|item| { item.parse::<u64>().ok() }) |                 .filter_map(|item| { item.parse::<u64>().ok() }) | ||||||
| @@ -31,7 +35,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("PATREON_GUILD_ID").map(|var| var.parse::<u64>().ok()).ok().flatten(); |         env::var("CNC_GUILD").map(|var| var.parse::<u64>().ok()).ok().flatten(); | ||||||
|     pub static ref MIN_INTERVAL: i64 = |     pub static ref MIN_INTERVAL: i64 = | ||||||
|         env::var("MIN_INTERVAL").ok().and_then(|inner| inner.parse::<i64>().ok()).unwrap_or(600); |         env::var("MIN_INTERVAL").ok().and_then(|inner| inner.parse::<i64>().ok()).unwrap_or(600); | ||||||
|     pub static ref MAX_TIME: i64 = env::var("MAX_TIME") |     pub static ref MAX_TIME: i64 = env::var("MAX_TIME") | ||||||
| @@ -44,5 +48,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(|_| "/usr/bin/python3".to_string()); |         env::var("PYTHON_LOCATION").unwrap_or_else(|_| "venv/bin/python3".to_string()); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,7 +6,9 @@ use poise::{ | |||||||
|     serenity_prelude::{model::application::interaction::Interaction, utils::shard_id}, |     serenity_prelude::{model::application::interaction::Interaction, utils::shard_id}, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| use crate::{component_models::ComponentDataModel, Data, Error, THEME_COLOR}; | use crate::{ | ||||||
|  |     component_models::ComponentDataModel, models::guild_data::GuildData, Data, Error, THEME_COLOR, | ||||||
|  | }; | ||||||
|  |  | ||||||
| pub async fn listener( | pub async fn listener( | ||||||
|     ctx: &serenity::Context, |     ctx: &serenity::Context, | ||||||
| @@ -27,7 +29,7 @@ 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 IGNORE INTO guilds (guild) VALUES (?)", guild_id) |                 sqlx::query!("INSERT IGNORE INTO guilds (id) VALUES (?)", guild_id) | ||||||
|                     .execute(&data.database) |                     .execute(&data.database) | ||||||
|                     .await?; |                     .await?; | ||||||
|  |  | ||||||
| @@ -61,16 +63,28 @@ To stay up to date on the latest features and fixes, join our [Discord](https:// | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         poise::Event::GuildDelete { incomplete, .. } => { |         poise::Event::GuildDelete { incomplete, .. } => { | ||||||
|             let _ = sqlx::query!("DELETE FROM guilds WHERE guild = ?", incomplete.id.0) |             let _ = sqlx::query!("DELETE FROM guilds WHERE id = ?", incomplete.id.0) | ||||||
|                 .execute(&data.database) |                 .execute(&data.database) | ||||||
|                 .await; |                 .await; | ||||||
|         } |         } | ||||||
|         poise::Event::InteractionCreate { interaction } => { |         poise::Event::InteractionCreate { interaction } => { | ||||||
|             if let Interaction::MessageComponent(component) = interaction { |             match interaction { | ||||||
|                 let component_model = ComponentDataModel::from_custom_id(&component.data.custom_id); |                 Interaction::ApplicationCommand(app_command) => { | ||||||
|  |                     if let Some(guild_id) = app_command.guild_id { | ||||||
|  |                         // check database guild exists | ||||||
|  |                         GuildData::from_guild(guild_id, &data.database).await?; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 Interaction::MessageComponent(component) => { | ||||||
|  |                     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; | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|  |                 _ => {} | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|         _ => {} |         _ => {} | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -110,14 +110,13 @@ 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, u64), |     current: (u64, u64, u64), | ||||||
| } | } | ||||||
|  |  | ||||||
| impl<'a> Parser<'a> { | impl<'a> Parser<'a> { | ||||||
| @@ -141,17 +140,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 day, mut sec, nsec) = match &self.src[start..end] { |         let (mut month, mut sec, nsec) = match &self.src[start..end] { | ||||||
|             "nanos" | "nsec" | "ns" => (0, 0u64, 0u64, n), |             "nanos" | "nsec" | "ns" => (0u64, 0u64, n), | ||||||
|             "usec" | "us" => (0, 0, 0u64, n.mul(1000)?), |             "usec" | "us" => (0, 0u64, n.mul(1000)?), | ||||||
|             "millis" | "msec" | "ms" => (0, 0, 0u64, n.mul(1_000_000)?), |             "millis" | "msec" | "ms" => (0, 0u64, n.mul(1_000_000)?), | ||||||
|             "seconds" | "second" | "secs" | "sec" | "s" => (0, 0, n, 0), |             "seconds" | "second" | "secs" | "sec" | "s" => (0, n, 0), | ||||||
|             "minutes" | "minute" | "min" | "mins" | "m" => (0, 0, n.mul(60)?, 0), |             "minutes" | "minute" | "min" | "mins" | "m" => (0, n.mul(60)?, 0), | ||||||
|             "hours" | "hour" | "hr" | "hrs" | "h" => (0, 0, n.mul(3600)?, 0), |             "hours" | "hour" | "hr" | "hrs" | "h" => (0, n.mul(3600)?, 0), | ||||||
|             "days" | "day" | "d" => (0, n, 0, 0), |             "days" | "day" | "d" => (0, n.mul(86400)?, 0), | ||||||
|             "weeks" | "week" | "w" => (0, n.mul(7)?, 0, 0), |             "weeks" | "week" | "w" => (0, n.mul(86400 * 7)?, 0), | ||||||
|             "months" | "month" | "M" => (n, 0, 0, 0), |             "months" | "month" | "M" => (n, 0, 0), | ||||||
|             "years" | "year" | "y" => (n.mul(12)?, 0, 0, 0), |             "years" | "year" | "y" => (12, 0, 0), | ||||||
|             _ => { |             _ => { | ||||||
|                 return Err(Error::UnknownUnit { |                 return Err(Error::UnknownUnit { | ||||||
|                     start, |                     start, | ||||||
| @@ -161,16 +160,15 @@ impl<'a> Parser<'a> { | |||||||
|                 }); |                 }); | ||||||
|             } |             } | ||||||
|         }; |         }; | ||||||
|         let mut nsec = self.current.3 + nsec; |         let mut nsec = self.current.2 + nsec; | ||||||
|         if nsec > 1_000_000_000 { |         if nsec > 1_000_000_000 { | ||||||
|             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.2; |         sec += self.current.1; | ||||||
|         day += self.current.1; |  | ||||||
|         month += self.current.0; |         month += self.current.0; | ||||||
|  |  | ||||||
|         self.current = (month, day, sec, nsec); |         self.current = (month, sec, nsec); | ||||||
|  |  | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
| @@ -217,13 +215,7 @@ 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 => { |                 None => return Ok(Interval { month: self.current.0, sec: self.current.1 }), | ||||||
|                     return Ok(Interval { |  | ||||||
|                         month: self.current.0, |  | ||||||
|                         day: self.current.1, |  | ||||||
|                         sec: self.current.2, |  | ||||||
|                     }) |  | ||||||
|                 } |  | ||||||
|             }; |             }; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -255,73 +247,5 @@ 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, 0) }.parse() |     Parser { iter: s.chars(), src: s, current: (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); |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										15
									
								
								src/main.rs
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								src/main.rs
									
									
									
									
									
								
							| @@ -18,10 +18,10 @@ use std::{ | |||||||
|     env, |     env, | ||||||
|     error::Error as StdError, |     error::Error as StdError, | ||||||
|     fmt::{Debug, Display, Formatter}, |     fmt::{Debug, Display, Formatter}, | ||||||
|     path::Path, |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
|  | use dotenv::dotenv; | ||||||
| use log::{error, warn}; | use log::{error, warn}; | ||||||
| use poise::serenity_prelude::model::{ | use poise::serenity_prelude::model::{ | ||||||
|     gateway::GatewayIntents, |     gateway::GatewayIntents, | ||||||
| @@ -75,7 +75,7 @@ impl Display for Ended { | |||||||
|  |  | ||||||
| impl StdError for Ended {} | impl StdError for Ended {} | ||||||
|  |  | ||||||
| #[tokio::main(flavor = "multi_thread")] | #[tokio::main] | ||||||
| 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); | ||||||
|  |  | ||||||
| @@ -88,9 +88,7 @@ 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(); | ||||||
|  |  | ||||||
|     if Path::new("/etc/reminder-rs/config.env").exists() { |     dotenv()?; | ||||||
|         dotenv::from_path("/etc/reminder-rs/config.env")?; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     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"); | ||||||
|  |  | ||||||
| @@ -122,6 +120,10 @@ async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | |||||||
|                 ], |                 ], | ||||||
|                 ..command_macro::macro_base() |                 ..command_macro::macro_base() | ||||||
|             }, |             }, | ||||||
|  |             poise::Command { | ||||||
|  |                 subcommands: vec![moderation_cmds::default_channel()], | ||||||
|  |                 ..moderation_cmds::default() | ||||||
|  |             }, | ||||||
|             reminder_cmds::pause(), |             reminder_cmds::pause(), | ||||||
|             reminder_cmds::offset(), |             reminder_cmds::offset(), | ||||||
|             reminder_cmds::nudge(), |             reminder_cmds::nudge(), | ||||||
| @@ -135,7 +137,6 @@ 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![ | ||||||
| @@ -170,8 +171,6 @@ async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | |||||||
|     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 IFNULL(timezone, 'UTC') AS timezone |         "SELECT IFNULL(timezone, 'UTC') AS timezone | ||||||
|         FROM users |         FROM users | ||||||
|   | |||||||
| @@ -38,7 +38,7 @@ SELECT id, name, nudge, blacklisted, webhook_id, webhook_token, paused, paused_u | |||||||
|  |  | ||||||
|             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 (?, ?, ?) | ||||||
|                 ", |                 ", | ||||||
|                 channel_id, |                 channel_id, | ||||||
|                 channel_name, |                 channel_name, | ||||||
|   | |||||||
| @@ -43,7 +43,7 @@ pub async fn guild_command_macro( | |||||||
| ) -> Option<CommandMacro<Data, Error>> { | ) -> Option<CommandMacro<Data, Error>> { | ||||||
|     let row = sqlx::query!( |     let row = sqlx::query!( | ||||||
|         " |         " | ||||||
| SELECT * FROM macro WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?) AND name = ? | SELECT * FROM macro WHERE guild_id = ? AND name = ? | ||||||
|         ", |         ", | ||||||
|         ctx.guild_id().unwrap().0, |         ctx.guild_id().unwrap().0, | ||||||
|         name |         name | ||||||
|   | |||||||
							
								
								
									
										52
									
								
								src/models/guild_data.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								src/models/guild_data.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | |||||||
|  | use sqlx::MySqlPool; | ||||||
|  |  | ||||||
|  | use crate::GuildId; | ||||||
|  |  | ||||||
|  | pub struct GuildData { | ||||||
|  |     pub id: u64, | ||||||
|  |     pub default_channel: Option<u64>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl GuildData { | ||||||
|  |     pub async fn from_guild(guild: GuildId, pool: &MySqlPool) -> Result<Self, sqlx::Error> { | ||||||
|  |         let guild_id = guild.0; | ||||||
|  |  | ||||||
|  |         if let Ok(row) = sqlx::query_as_unchecked!( | ||||||
|  |             Self, | ||||||
|  |             " | ||||||
|  | SELECT id, default_channel FROM guilds WHERE id = ? | ||||||
|  |             ", | ||||||
|  |             guild_id | ||||||
|  |         ) | ||||||
|  |         .fetch_one(pool) | ||||||
|  |         .await | ||||||
|  |         { | ||||||
|  |             Ok(row) | ||||||
|  |         } else { | ||||||
|  |             sqlx::query!( | ||||||
|  |                 " | ||||||
|  | INSERT IGNORE INTO guilds (id) VALUES (?) | ||||||
|  |                 ", | ||||||
|  |                 guild_id | ||||||
|  |             ) | ||||||
|  |             .execute(&pool.clone()) | ||||||
|  |             .await?; | ||||||
|  |  | ||||||
|  |             Ok(Self { id: guild_id, default_channel: None }) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pub async fn commit_changes(&self, pool: &MySqlPool) -> Result<(), sqlx::Error> { | ||||||
|  |         sqlx::query!( | ||||||
|  |             " | ||||||
|  | UPDATE guilds SET default_channel = ? WHERE id = ? | ||||||
|  |             ", | ||||||
|  |             self.default_channel, | ||||||
|  |             self.id | ||||||
|  |         ) | ||||||
|  |         .execute(pool) | ||||||
|  |         .await?; | ||||||
|  |  | ||||||
|  |         Ok(()) | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,28 +1,28 @@ | |||||||
| 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_prelude::{async_trait, model::id::UserId}; | use log::warn; | ||||||
|  | use poise::serenity_prelude::{async_trait, model::id::UserId, ChannelId}; | ||||||
|  |  | ||||||
| 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, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| #[async_trait] | #[async_trait] | ||||||
| pub trait CtxData { | pub trait CtxData { | ||||||
|     async fn user_data<U: Into<UserId> + Send>(&self, user_id: U) -> Result<UserData, Error>; |     async fn user_data<U: Into<UserId> + Send>(&self, user_id: U) -> Result<UserData, Error>; | ||||||
|  |  | ||||||
|     async fn author_data(&self) -> Result<UserData, Error>; |     async fn author_data(&self) -> Result<UserData, 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>; | ||||||
|  |     async fn guild_data(&self) -> Option<GuildData>; | ||||||
|     async fn command_macros(&self) -> Result<Vec<CommandMacro<Data, Error>>, Error>; |     async fn command_macros(&self) -> Result<Vec<CommandMacro<Data, Error>>, Error>; | ||||||
|  |     async fn default_channel(&self) -> Option<ChannelId>; | ||||||
| } | } | ||||||
|  |  | ||||||
| #[async_trait] | #[async_trait] | ||||||
| @@ -51,24 +51,55 @@ impl CtxData for Context<'_> { | |||||||
|     async fn command_macros(&self) -> Result<Vec<CommandMacro<Data, Error>>, Error> { |     async fn command_macros(&self) -> Result<Vec<CommandMacro<Data, Error>>, Error> { | ||||||
|         self.data().command_macros(self.guild_id().unwrap()).await |         self.data().command_macros(self.guild_id().unwrap()).await | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     async fn default_channel(&self) -> Option<ChannelId> { | ||||||
|  |         match self.guild_id() { | ||||||
|  |             Some(guild_id) => { | ||||||
|  |                 let guild_data = GuildData::from_guild(guild_id, &self.data().database).await; | ||||||
|  |  | ||||||
|  |                 match guild_data { | ||||||
|  |                     Ok(data) => data.default_channel.map(|c| ChannelId(c)), | ||||||
|  |  | ||||||
|  |                     Err(e) => { | ||||||
|  |                         warn!("SQL error: {:?}", e); | ||||||
|  |  | ||||||
|  |                         None | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             None => None, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async fn guild_data(&self) -> Option<GuildData> { | ||||||
|  |         match self.guild_id() { | ||||||
|  |             Some(guild_id) => GuildData::from_guild(guild_id, &self.data().database).await.ok(), | ||||||
|  |  | ||||||
|  |             None => None, | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| impl Data { | impl Data { | ||||||
|     pub(crate) async fn command_macros( |     pub async fn command_macros( | ||||||
|         &self, |         &self, | ||||||
|         guild_id: GuildId, |         guild_id: GuildId, | ||||||
|     ) -> Result<Vec<CommandMacro<Data, Error>>, Error> { |     ) -> Result<Vec<CommandMacro<Data, Error>>, Error> { | ||||||
|         let rows = sqlx::query!( |         let rows = sqlx::query!( | ||||||
|             "SELECT name, description, commands FROM macro WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?)", |             "SELECT name, description, commands FROM macro WHERE guild_id = ?", | ||||||
|             guild_id.0 |             guild_id.0 | ||||||
|         ) |         ) | ||||||
|         .fetch_all(&self.database) |         .fetch_all(&self.database) | ||||||
|         .await?.iter().map(|row| CommandMacro { |         .await? | ||||||
|  |         .iter() | ||||||
|  |         .map(|row| CommandMacro { | ||||||
|             guild_id, |             guild_id, | ||||||
|             name: row.name.clone(), |             name: row.name.clone(), | ||||||
|             description: row.description.clone(), |             description: row.description.clone(), | ||||||
|             commands: serde_json::from_str(&row.commands).unwrap(), |             commands: serde_json::from_str(&row.commands).unwrap(), | ||||||
|         }).collect(); |         }) | ||||||
|  |         .collect(); | ||||||
|  |  | ||||||
|         Ok(rows) |         Ok(rows) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -53,8 +53,7 @@ pub struct ReminderBuilder { | |||||||
|     channel: u32, |     channel: u32, | ||||||
|     utc_time: NaiveDateTime, |     utc_time: NaiveDateTime, | ||||||
|     timezone: String, |     timezone: String, | ||||||
|     interval_seconds: Option<i64>, |     interval_secs: Option<i64>, | ||||||
|     interval_days: Option<i64>, |  | ||||||
|     interval_months: Option<i64>, |     interval_months: Option<i64>, | ||||||
|     expires: Option<NaiveDateTime>, |     expires: Option<NaiveDateTime>, | ||||||
|     content: String, |     content: String, | ||||||
| @@ -88,7 +87,6 @@ INSERT INTO reminders ( | |||||||
|     `utc_time`, |     `utc_time`, | ||||||
|     `timezone`, |     `timezone`, | ||||||
|     `interval_seconds`, |     `interval_seconds`, | ||||||
|     `interval_days`, |  | ||||||
|     `interval_months`, |     `interval_months`, | ||||||
|     `expires`, |     `expires`, | ||||||
|     `content`, |     `content`, | ||||||
| @@ -108,7 +106,6 @@ INSERT INTO reminders ( | |||||||
|     ?, |     ?, | ||||||
|     ?, |     ?, | ||||||
|     ?, |     ?, | ||||||
|     ?, |  | ||||||
|     ? |     ? | ||||||
| ) | ) | ||||||
|             ", |             ", | ||||||
| @@ -116,8 +113,7 @@ INSERT INTO reminders ( | |||||||
|                         self.channel, |                         self.channel, | ||||||
|                         utc_time, |                         utc_time, | ||||||
|                         self.timezone, |                         self.timezone, | ||||||
|                         self.interval_seconds, |                         self.interval_secs, | ||||||
|                         self.interval_days, |  | ||||||
|                         self.interval_months, |                         self.interval_months, | ||||||
|                         self.expires, |                         self.expires, | ||||||
|                         self.content, |                         self.content, | ||||||
| @@ -179,15 +175,17 @@ 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 { | ||||||
|         if let Some(utc_time) = NaiveDateTime::from_timestamp_opt(time.into(), 0) { |         self.utc_time = NaiveDateTime::from_timestamp(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 { | ||||||
|         self.expires = time.map(|t| NaiveDateTime::from_timestamp_opt(t.into(), 0)).flatten(); |         if let Some(t) = time { | ||||||
|  |             self.expires = Some(NaiveDateTime::from_timestamp(t.into(), 0)); | ||||||
|  |         } else { | ||||||
|  |             self.expires = None; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         self |         self | ||||||
|     } |     } | ||||||
| @@ -214,14 +212,9 @@ impl<'a> MultiReminderBuilder<'a> { | |||||||
|  |  | ||||||
|         let mut ok_locs = HashSet::new(); |         let mut ok_locs = HashSet::new(); | ||||||
|  |  | ||||||
|         if self |         if self.interval.map_or(false, |i| ((i.sec + i.month * 30 * DAY) as i64) < *MIN_INTERVAL) { | ||||||
|             .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 |         } else if self.interval.map_or(false, |i| ((i.sec + i.month * 30 * DAY) as i64) > *MAX_TIME) | ||||||
|             .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 { | ||||||
| @@ -309,8 +302,7 @@ impl<'a> MultiReminderBuilder<'a> { | |||||||
|                             channel: c, |                             channel: c, | ||||||
|                             utc_time: self.utc_time, |                             utc_time: self.utc_time, | ||||||
|                             timezone: self.timezone.to_string(), |                             timezone: self.timezone.to_string(), | ||||||
|                             interval_seconds: self.interval.map(|i| i.sec as i64), |                             interval_secs: 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(), | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ pub mod look_flags; | |||||||
|  |  | ||||||
| use std::hash::{Hash, Hasher}; | use std::hash::{Hash, Hasher}; | ||||||
|  |  | ||||||
| use chrono::{DateTime, NaiveDateTime, Utc}; | use chrono::{NaiveDateTime, TimeZone}; | ||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
| use poise::serenity_prelude::{ | use poise::serenity_prelude::{ | ||||||
|     model::id::{ChannelId, GuildId, UserId}, |     model::id::{ChannelId, GuildId, UserId}, | ||||||
| @@ -24,9 +24,8 @@ pub struct Reminder { | |||||||
|     pub id: u32, |     pub id: u32, | ||||||
|     pub uid: String, |     pub uid: String, | ||||||
|     pub channel: u64, |     pub channel: u64, | ||||||
|     pub utc_time: DateTime<Utc>, |     pub utc_time: NaiveDateTime, | ||||||
|     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, | ||||||
| @@ -60,7 +59,6 @@ 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, | ||||||
| @@ -97,7 +95,6 @@ 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, | ||||||
| @@ -141,7 +138,6 @@ 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, | ||||||
| @@ -199,7 +195,6 @@ 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, | ||||||
| @@ -233,7 +228,6 @@ 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, | ||||||
| @@ -251,7 +245,7 @@ LEFT JOIN | |||||||
| ON | ON | ||||||
|     reminders.set_by = users.id |     reminders.set_by = users.id | ||||||
| WHERE | WHERE | ||||||
|     channels.guild_id = (SELECT id FROM guilds WHERE guild = ?) |     channels.guild_id = ? | ||||||
|                 ", |                 ", | ||||||
|                     guild_id.as_u64() |                     guild_id.as_u64() | ||||||
|                 ) |                 ) | ||||||
| @@ -268,7 +262,6 @@ 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, | ||||||
| @@ -317,32 +310,30 @@ WHERE | |||||||
|             count + 1, |             count + 1, | ||||||
|             self.display_content(), |             self.display_content(), | ||||||
|             self.channel, |             self.channel, | ||||||
|             self.utc_time.with_timezone(timezone).format("%Y-%m-%d %H:%M:%S") |             timezone.timestamp(self.utc_time.timestamp(), 0).format("%Y-%m-%d %H:%M:%S") | ||||||
|         ) |         ) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     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 => { |             TimeDisplayType::Absolute => timezone | ||||||
|                 self.utc_time.with_timezone(timezone).format("%Y-%m-%d %H:%M:%S").to_string() |                 .timestamp(self.utc_time.timestamp(), 0) | ||||||
|             } |                 .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() |         if self.interval_seconds.is_some() || self.interval_months.is_some() { | ||||||
|             || self.interval_days.is_some() |  | ||||||
|             || self.interval_months.is_some() |  | ||||||
|         { |  | ||||||
|             format!( |             format!( | ||||||
|                 "'{}' *occurs next at* **{}**, repeating (set by {})\n", |                 "'{}' *occurs next at* **{}**, repeating (set by {})", | ||||||
|                 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 {})\n", |                 "'{}' *occurs next at* **{}** (set by {})", | ||||||
|                 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::{DateTime, Utc}; | use chrono::NaiveDateTime; | ||||||
| use sqlx::MySqlPool; | use sqlx::MySqlPool; | ||||||
|  |  | ||||||
| pub struct Timer { | pub struct Timer { | ||||||
|     pub name: String, |     pub name: String, | ||||||
|     pub start_time: DateTime<Utc>, |     pub start_time: NaiveDateTime, | ||||||
|     pub owner: u64, |     pub owner: u64, | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,13 +0,0 @@ | |||||||
| [Unit] |  | ||||||
| Description=Reminder Bot |  | ||||||
|  |  | ||||||
| [Service] |  | ||||||
| User=reminder |  | ||||||
| Type=simple |  | ||||||
| ExecStart=/usr/bin/reminder-rs |  | ||||||
| Restart=always |  | ||||||
| RestartSec=4 |  | ||||||
| # Environment="RUST_LOG=warn,reminder_rs=info,postman=info" |  | ||||||
|  |  | ||||||
| [Install] |  | ||||||
| WantedBy=multi-user.target |  | ||||||
| @@ -31,7 +31,7 @@ lazy_static! { | |||||||
|     ) |     ) | ||||||
|         .into(); |         .into(); | ||||||
|     pub static ref SUBSCRIPTION_ROLES: HashSet<u64> = HashSet::from_iter( |     pub static ref SUBSCRIPTION_ROLES: HashSet<u64> = HashSet::from_iter( | ||||||
|         env::var("PATREON_ROLE_ID") |         env::var("SUBSCRIPTION_ROLES") | ||||||
|             .map(|var| var |             .map(|var| var | ||||||
|                 .split(',') |                 .split(',') | ||||||
|                 .filter_map(|item| { item.parse::<u64>().ok() }) |                 .filter_map(|item| { item.parse::<u64>().ok() }) | ||||||
| @@ -39,7 +39,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("PATREON_GUILD_ID").map(|var| var.parse::<u64>().ok()).ok().flatten(); |         env::var("CNC_GUILD").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()) | ||||||
|   | |||||||
| @@ -75,7 +75,7 @@ pub async fn initialize( | |||||||
|     env::var("OAUTH2_CLIENT_ID").expect("`OAUTH2_CLIENT_ID' not supplied"); |     env::var("OAUTH2_CLIENT_ID").expect("`OAUTH2_CLIENT_ID' not supplied"); | ||||||
|     env::var("OAUTH2_CLIENT_SECRET").expect("`OAUTH2_CLIENT_SECRET' 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("OAUTH2_DISCORD_CALLBACK").expect("`OAUTH2_DISCORD_CALLBACK' not supplied"); | ||||||
|     env::var("PATREON_GUILD_ID").expect("`PATREON_GUILD_ID' not supplied"); |     env::var("PATREON_GUILD_ID").expect("`PATREON_GUILD' not supplied"); | ||||||
|     info!("Done!"); |     info!("Done!"); | ||||||
|  |  | ||||||
|     let oauth2_client = BasicClient::new( |     let oauth2_client = BasicClient::new( | ||||||
|   | |||||||
| @@ -58,7 +58,6 @@ pub async fn export_reminders( | |||||||
|                  reminders.enabled, |                  reminders.enabled, | ||||||
|                  reminders.expires, |                  reminders.expires, | ||||||
|                  reminders.interval_seconds, |                  reminders.interval_seconds, | ||||||
|                  reminders.interval_days, |  | ||||||
|                  reminders.interval_months, |                  reminders.interval_months, | ||||||
|                  reminders.name, |                  reminders.name, | ||||||
|                  reminders.restartable, |                  reminders.restartable, | ||||||
| @@ -160,7 +159,6 @@ pub async fn import_reminders( | |||||||
|                                     enabled: record.enabled, |                                     enabled: record.enabled, | ||||||
|                                     expires: record.expires, |                                     expires: record.expires, | ||||||
|                                     interval_seconds: record.interval_seconds, |                                     interval_seconds: record.interval_seconds, | ||||||
|                                     interval_days: record.interval_days, |  | ||||||
|                                     interval_months: record.interval_months, |                                     interval_months: record.interval_months, | ||||||
|                                     name: record.name, |                                     name: record.name, | ||||||
|                                     restartable: record.restartable, |                                     restartable: record.restartable, | ||||||
| @@ -320,6 +318,13 @@ pub async fn import_todos( | |||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|  |                 let _ = sqlx::query!( | ||||||
|  |                     "DELETE FROM todos WHERE guild_id = (SELECT id FROM guilds WHERE guild = ?)", | ||||||
|  |                     id | ||||||
|  |                 ) | ||||||
|  |                 .execute(pool.inner()) | ||||||
|  |                 .await; | ||||||
|  |  | ||||||
|                 let query_str = format!( |                 let query_str = format!( | ||||||
|                     "INSERT INTO todos (value, channel_id, guild_id) VALUES {}", |                     "INSERT INTO todos (value, channel_id, guild_id) VALUES {}", | ||||||
|                     vec![query_placeholder].repeat(query_params.len()).join(",") |                     vec![query_placeholder].repeat(query_params.len()).join(",") | ||||||
|   | |||||||
| @@ -16,12 +16,10 @@ use serenity::{ | |||||||
| use sqlx::{MySql, Pool}; | use sqlx::{MySql, Pool}; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
|     check_guild_subscription, check_subscription, |  | ||||||
|     consts::{ |     consts::{ | ||||||
|         MAX_CONTENT_LENGTH, MAX_EMBED_AUTHOR_LENGTH, MAX_EMBED_DESCRIPTION_LENGTH, |         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_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, |         MAX_EMBED_FOOTER_LENGTH, MAX_EMBED_TITLE_LENGTH, MAX_URL_LENGTH, MAX_USERNAME_LENGTH, | ||||||
|         MIN_INTERVAL, |  | ||||||
|     }, |     }, | ||||||
|     routes::dashboard::{ |     routes::dashboard::{ | ||||||
|         create_database_channel, create_reminder, template_name_default, DeleteReminder, |         create_database_channel, create_reminder, template_name_default, DeleteReminder, | ||||||
| @@ -249,9 +247,9 @@ pub async fn create_reminder_template( | |||||||
|             Ok(json!({})) |             Ok(json!({})) | ||||||
|         } |         } | ||||||
|         Err(e) => { |         Err(e) => { | ||||||
|             warn!("Could not create template for {}: {:?}", id, e); |             warn!("Could not fetch templates from {}: {:?}", id, e); | ||||||
|  |  | ||||||
|             json_err!("Could not create template") |             json_err!("Could not get templates") | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -341,7 +339,6 @@ pub async fn get_reminders(id: u64, ctx: &State<Context>, pool: &State<Pool<MySq | |||||||
|                  reminders.enabled, |                  reminders.enabled, | ||||||
|                  reminders.expires, |                  reminders.expires, | ||||||
|                  reminders.interval_seconds, |                  reminders.interval_seconds, | ||||||
|                  reminders.interval_days, |  | ||||||
|                  reminders.interval_months, |                  reminders.interval_months, | ||||||
|                  reminders.name, |                  reminders.name, | ||||||
|                  reminders.restartable, |                  reminders.restartable, | ||||||
| @@ -377,109 +374,35 @@ pub async fn edit_reminder( | |||||||
|     reminder: Json<PatchReminder>, |     reminder: Json<PatchReminder>, | ||||||
|     serenity_context: &State<Context>, |     serenity_context: &State<Context>, | ||||||
|     pool: &State<Pool<MySql>>, |     pool: &State<Pool<MySql>>, | ||||||
|     cookies: &CookieJar<'_>, |  | ||||||
| ) -> JsonResult { | ) -> JsonResult { | ||||||
|     check_authorization!(cookies, serenity_context.inner(), id); |  | ||||||
|  |  | ||||||
|     let mut error = vec![]; |     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!(pool.inner(), error, reminder.[ |  | ||||||
|             content, |  | ||||||
|             embed_author, |  | ||||||
|             embed_description, |  | ||||||
|             embed_footer, |  | ||||||
|             embed_title, |  | ||||||
|             embed_fields, |  | ||||||
|             username |  | ||||||
|         ]); |  | ||||||
|     } else { |  | ||||||
|         error.push("Message exceeds limits.".to_string()); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     update_field!(pool.inner(), error, reminder.[ |     update_field!(pool.inner(), error, reminder.[ | ||||||
|         attachment, |         attachment, | ||||||
|         attachment_name, |         attachment_name, | ||||||
|         avatar, |         avatar, | ||||||
|  |         content, | ||||||
|  |         embed_author, | ||||||
|         embed_author_url, |         embed_author_url, | ||||||
|         embed_color, |         embed_color, | ||||||
|  |         embed_description, | ||||||
|  |         embed_footer, | ||||||
|         embed_footer_url, |         embed_footer_url, | ||||||
|         embed_image_url, |         embed_image_url, | ||||||
|         embed_thumbnail_url, |         embed_thumbnail_url, | ||||||
|  |         embed_title, | ||||||
|  |         embed_fields, | ||||||
|         enabled, |         enabled, | ||||||
|         expires, |         expires, | ||||||
|  |         interval_seconds, | ||||||
|  |         interval_months, | ||||||
|         name, |         name, | ||||||
|         restartable, |         restartable, | ||||||
|         tts, |         tts, | ||||||
|  |         username, | ||||||
|         utc_time |         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(&serenity_context.inner(), id).await |  | ||||||
|             || check_subscription(&serenity_context.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(pool.inner()) |  | ||||||
|                 .await |  | ||||||
|                 .map_err(|e| { |  | ||||||
|                     warn!("Error updating reminder interval: {:?}", e); |  | ||||||
|                     json!({ "reminder": Option::<Reminder>::None, "errors": vec!["Unknown error"] }) |  | ||||||
|                 })? |  | ||||||
|                 .days |  | ||||||
|                 .unwrap_or(0), |  | ||||||
|             } + 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(pool.inner()) |  | ||||||
|                 .await |  | ||||||
|                 .map_err(|e| { |  | ||||||
|                     warn!("Error updating reminder interval: {:?}", e); |  | ||||||
|                     json!({ "reminder": Option::<Reminder>::None, "errors": vec!["Unknown error"] }) |  | ||||||
|                 })? |  | ||||||
|                 .months |  | ||||||
|                 .unwrap_or(0), |  | ||||||
|             } + 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(pool.inner()) |  | ||||||
|                 .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!(pool.inner(), error, reminder.[ |  | ||||||
|                     interval_days, |  | ||||||
|                     interval_months, |  | ||||||
|                     interval_seconds |  | ||||||
|                 ]); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if reminder.channel > 0 { |     if reminder.channel > 0 { | ||||||
|         let channel = ChannelId(reminder.channel).to_channel_cached(&serenity_context.inner()); |         let channel = ChannelId(reminder.channel).to_channel_cached(&serenity_context.inner()); | ||||||
|         match channel { |         match channel { | ||||||
| @@ -560,7 +483,6 @@ pub async fn edit_reminder( | |||||||
|          reminders.enabled, |          reminders.enabled, | ||||||
|          reminders.expires, |          reminders.expires, | ||||||
|          reminders.interval_seconds, |          reminders.interval_seconds, | ||||||
|          reminders.interval_days, |  | ||||||
|          reminders.interval_months, |          reminders.interval_months, | ||||||
|          reminders.name, |          reminders.name, | ||||||
|          reminders.restartable, |          reminders.restartable, | ||||||
|   | |||||||
| @@ -8,13 +8,13 @@ use rocket::{ | |||||||
|     serde::json::{json, Value as JsonValue}, |     serde::json::{json, Value as JsonValue}, | ||||||
| }; | }; | ||||||
| use rocket_dyn_templates::Template; | use rocket_dyn_templates::Template; | ||||||
| use serde::{Deserialize, Deserializer, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
| use serenity::{ | use serenity::{ | ||||||
|     client::Context, |     client::Context, | ||||||
|     http::Http, |     http::Http, | ||||||
|     model::id::{ChannelId, GuildId, UserId}, |     model::id::{ChannelId, GuildId, UserId}, | ||||||
| }; | }; | ||||||
| use sqlx::{types::Json, Executor}; | use sqlx::{types::Json, Executor, MySql, Pool}; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
|     check_guild_subscription, check_subscription, |     check_guild_subscription, check_subscription, | ||||||
| @@ -50,18 +50,6 @@ 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")] | ||||||
| @@ -144,7 +132,6 @@ 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, | ||||||
| @@ -177,7 +164,6 @@ pub struct ReminderCsv { | |||||||
|     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, | ||||||
| @@ -191,13 +177,10 @@ pub struct ReminderCsv { | |||||||
| 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")] | ||||||
| @@ -207,7 +190,6 @@ 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>, | ||||||
| @@ -216,13 +198,10 @@ 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>, | ||||||
| @@ -231,16 +210,10 @@ 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 = "interval_default")] |     #[serde(default)] | ||||||
|     #[serde(deserialize_with = "deserialize_optional_field")] |  | ||||||
|     interval_seconds: Unset<Option<u32>>, |     interval_seconds: Unset<Option<u32>>, | ||||||
|     #[serde(default = "interval_default")] |     #[serde(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>, | ||||||
| @@ -249,36 +222,11 @@ pub struct PatchReminder { | |||||||
|     #[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(); | ||||||
|  |  | ||||||
| @@ -353,28 +301,11 @@ pub struct TodoCsv { | |||||||
|  |  | ||||||
| pub async fn create_reminder( | pub async fn create_reminder( | ||||||
|     ctx: &Context, |     ctx: &Context, | ||||||
|     pool: impl sqlx::Executor<'_, Database = Database> + Copy, |     pool: &Pool<MySql>, | ||||||
|     guild_id: GuildId, |     guild_id: GuildId, | ||||||
|     user_id: UserId, |     user_id: UserId, | ||||||
|     reminder: Reminder, |     reminder: Reminder, | ||||||
| ) -> JsonResult { | ) -> JsonResult { | ||||||
|     // check guild in db |  | ||||||
|     match sqlx::query!("SELECT 1 as A FROM guilds WHERE guild = ?", guild_id.0) |  | ||||||
|         .fetch_one(pool) |  | ||||||
|         .await |  | ||||||
|     { |  | ||||||
|         Err(sqlx::Error::RowNotFound) => { |  | ||||||
|             if sqlx::query!("INSERT INTO guilds (guild) VALUES (?)", guild_id.0) |  | ||||||
|                 .execute(pool) |  | ||||||
|                 .await |  | ||||||
|                 .is_err() |  | ||||||
|             { |  | ||||||
|                 return Err(json!({"error": "Guild could not be created"})); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         _ => {} |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // validate channel |     // validate channel | ||||||
|     let channel = ChannelId(reminder.channel).to_channel_cached(&ctx); |     let channel = ChannelId(reminder.channel).to_channel_cached(&ctx); | ||||||
|     let channel_exists = channel.is_some(); |     let channel_exists = channel.is_some(); | ||||||
| @@ -439,12 +370,8 @@ pub async fn create_reminder( | |||||||
|     if reminder.utc_time < Utc::now().naive_utc() { |     if reminder.utc_time < Utc::now().naive_utc() { | ||||||
|         return Err(json!({"error": "Time must be in the future"})); |         return Err(json!({"error": "Time must be in the future"})); | ||||||
|     } |     } | ||||||
|     if reminder.interval_seconds.is_some() |     if reminder.interval_seconds.is_some() || reminder.interval_months.is_some() { | ||||||
|         || reminder.interval_days.is_some() |  | ||||||
|         || reminder.interval_months.is_some() |  | ||||||
|     { |  | ||||||
|         if reminder.interval_months.unwrap_or(0) * 30 * DAY as u32 |         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) |             + reminder.interval_seconds.unwrap_or(0) | ||||||
|             < *MIN_INTERVAL |             < *MIN_INTERVAL | ||||||
|         { |         { | ||||||
| @@ -453,10 +380,7 @@ pub async fn create_reminder( | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     // check patreon if necessary |     // check patreon if necessary | ||||||
|     if reminder.interval_seconds.is_some() |     if reminder.interval_seconds.is_some() || reminder.interval_months.is_some() { | ||||||
|         || reminder.interval_days.is_some() |  | ||||||
|         || reminder.interval_months.is_some() |  | ||||||
|     { |  | ||||||
|         if !check_guild_subscription(&ctx, guild_id).await |         if !check_guild_subscription(&ctx, guild_id).await | ||||||
|             && !check_subscription(&ctx, user_id).await |             && !check_subscription(&ctx, user_id).await | ||||||
|         { |         { | ||||||
| @@ -467,11 +391,6 @@ pub async fn create_reminder( | |||||||
|     // base64 decode error dropped here |     // base64 decode error dropped here | ||||||
|     let attachment_data = reminder.attachment.as_ref().map(|s| base64::decode(s).ok()).flatten(); |     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 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(); |     let new_uid = generate_uid(); | ||||||
|  |  | ||||||
| @@ -497,14 +416,13 @@ pub async fn create_reminder( | |||||||
|          enabled, |          enabled, | ||||||
|          expires, |          expires, | ||||||
|          interval_seconds, |          interval_seconds, | ||||||
|          interval_days, |  | ||||||
|          interval_months, |          interval_months, | ||||||
|          name, |          name, | ||||||
|          restartable, |          restartable, | ||||||
|          tts, |          tts, | ||||||
|          username, |          username, | ||||||
|          `utc_time` |          `utc_time` | ||||||
|         ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)", |         ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)", | ||||||
|         new_uid, |         new_uid, | ||||||
|         attachment_data, |         attachment_data, | ||||||
|         reminder.attachment_name, |         reminder.attachment_name, | ||||||
| @@ -524,12 +442,11 @@ pub async fn create_reminder( | |||||||
|         reminder.enabled, |         reminder.enabled, | ||||||
|         reminder.expires, |         reminder.expires, | ||||||
|         reminder.interval_seconds, |         reminder.interval_seconds, | ||||||
|         reminder.interval_days, |  | ||||||
|         reminder.interval_months, |         reminder.interval_months, | ||||||
|         name, |         name, | ||||||
|         reminder.restartable, |         reminder.restartable, | ||||||
|         reminder.tts, |         reminder.tts, | ||||||
|         username, |         reminder.username, | ||||||
|         reminder.utc_time, |         reminder.utc_time, | ||||||
|     ) |     ) | ||||||
|     .execute(pool) |     .execute(pool) | ||||||
| @@ -556,7 +473,6 @@ pub async fn create_reminder( | |||||||
|              reminders.enabled, |              reminders.enabled, | ||||||
|              reminders.expires, |              reminders.expires, | ||||||
|              reminders.interval_seconds, |              reminders.interval_seconds, | ||||||
|              reminders.interval_days, |  | ||||||
|              reminders.interval_months, |              reminders.interval_months, | ||||||
|              reminders.name, |              reminders.name, | ||||||
|              reminders.restartable, |              reminders.restartable, | ||||||
|   | |||||||
| @@ -135,14 +135,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. The server may be misconfigured. Please retry or alert us in Discord.", |                         "Your login request was rejected", | ||||||
|                     )) |                     )) | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             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)")) |             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)")) | ||||||
|         } |         } | ||||||
|     } else { |     } else { | ||||||
|         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)")) |         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)")) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,8 +7,8 @@ function get_interval(element) { | |||||||
|  |  | ||||||
|     return { |     return { | ||||||
|         months: parseInt(months) || null, |         months: parseInt(months) || null, | ||||||
|         days: parseInt(days) || null, |  | ||||||
|         seconds: |         seconds: | ||||||
|  |             (parseInt(days) || 0) * 86400 + | ||||||
|                 (parseInt(hours) || 0) * 3600 + |                 (parseInt(hours) || 0) * 3600 + | ||||||
|                 (parseInt(minutes) || 0) * 60 + |                 (parseInt(minutes) || 0) * 60 + | ||||||
|                 (parseInt(seconds) || 0) || null, |                 (parseInt(seconds) || 0) || null, | ||||||
| @@ -22,15 +22,6 @@ function update_interval(element) { | |||||||
|     let minutes = element.querySelector('input[name="interval_minutes"]'); |     let minutes = element.querySelector('input[name="interval_minutes"]'); | ||||||
|     let seconds = element.querySelector('input[name="interval_seconds"]'); |     let seconds = element.querySelector('input[name="interval_seconds"]'); | ||||||
|  |  | ||||||
|     let interval = get_interval(element); |  | ||||||
|  |  | ||||||
|     if (interval.months === null && interval.days === null && interval.seconds === null) { |  | ||||||
|         months.value = ""; |  | ||||||
|         days.value = ""; |  | ||||||
|         hours.value = ""; |  | ||||||
|         minutes.value = ""; |  | ||||||
|         seconds.value = ""; |  | ||||||
|     } else { |  | ||||||
|     months.value = months.value.padStart(1, "0"); |     months.value = months.value.padStart(1, "0"); | ||||||
|     days.value = days.value.padStart(1, "0"); |     days.value = days.value.padStart(1, "0"); | ||||||
|     hours.value = hours.value.padStart(2, "0"); |     hours.value = hours.value.padStart(2, "0"); | ||||||
| @@ -42,10 +33,7 @@ function update_interval(element) { | |||||||
|         let remainder = seconds.value % 60; |         let remainder = seconds.value % 60; | ||||||
|  |  | ||||||
|         seconds.value = String(remainder).padStart(2, "0"); |         seconds.value = String(remainder).padStart(2, "0"); | ||||||
|             minutes.value = String(Number(minutes.value) + Number(quotient)).padStart( |         minutes.value = String(Number(minutes.value) + Number(quotient)).padStart(2, "0"); | ||||||
|                 2, |  | ||||||
|                 "0" |  | ||||||
|             ); |  | ||||||
|     } |     } | ||||||
|     if (minutes.value >= 60) { |     if (minutes.value >= 60) { | ||||||
|         let quotient = Math.floor(minutes.value / 60); |         let quotient = Math.floor(minutes.value / 60); | ||||||
| @@ -54,6 +42,12 @@ function update_interval(element) { | |||||||
|         minutes.value = String(remainder).padStart(2, "0"); |         minutes.value = String(remainder).padStart(2, "0"); | ||||||
|         hours.value = String(Number(hours.value) + Number(quotient)).padStart(2, "0"); |         hours.value = String(Number(hours.value) + Number(quotient)).padStart(2, "0"); | ||||||
|     } |     } | ||||||
|  |     if (hours.value >= 24) { | ||||||
|  |         let quotient = Math.floor(hours.value / 24); | ||||||
|  |         let remainder = hours.value % 24; | ||||||
|  |  | ||||||
|  |         hours.value = String(remainder).padStart(2, "0"); | ||||||
|  |         days.value = Number(days.value) + Number(quotient); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -60,15 +60,14 @@ function update_select(sel) { | |||||||
|         sel.closest("div.reminderContent").querySelector("img.discord-avatar").src = |         sel.closest("div.reminderContent").querySelector("img.discord-avatar").src = | ||||||
|             sel.selectedOptions[0].dataset["webhookAvatar"]; |             sel.selectedOptions[0].dataset["webhookAvatar"]; | ||||||
|     } else { |     } else { | ||||||
|         sel.closest("div.reminderContent").querySelector("img.discord-avatar").src = |         sel.closest("div.reminderContent").querySelector("img.discord-avatar").src = ""; | ||||||
|             "/static/img/icon.png"; |  | ||||||
|     } |     } | ||||||
|     if (sel.selectedOptions[0].dataset["webhookName"]) { |     if (sel.selectedOptions[0].dataset["webhookName"]) { | ||||||
|         sel.closest("div.reminderContent").querySelector("input.discord-username").value = |         sel.closest("div.reminderContent").querySelector("input.discord-username").value = | ||||||
|             sel.selectedOptions[0].dataset["webhookName"]; |             sel.selectedOptions[0].dataset["webhookName"]; | ||||||
|     } else { |     } else { | ||||||
|         sel.closest("div.reminderContent").querySelector("input.discord-username").value = |         sel.closest("div.reminderContent").querySelector("input.discord-username").value = | ||||||
|             "Reminder"; |             ""; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -320,7 +319,6 @@ async function serialize_reminder(node, mode) { | |||||||
|         embed_fields: fields, |         embed_fields: fields, | ||||||
|         expires: expiration_time, |         expires: expiration_time, | ||||||
|         interval_seconds: mode !== "template" ? interval.seconds : null, |         interval_seconds: mode !== "template" ? interval.seconds : null, | ||||||
|         interval_days: mode !== "template" ? interval.days : null, |  | ||||||
|         interval_months: mode !== "template" ? interval.months : null, |         interval_months: mode !== "template" ? interval.months : null, | ||||||
|         name: node.querySelector('input[name="name"]').value, |         name: node.querySelector('input[name="name"]').value, | ||||||
|         tts: node.querySelector('input[name="tts"]').checked, |         tts: node.querySelector('input[name="tts"]').checked, | ||||||
| @@ -333,9 +331,6 @@ function deserialize_reminder(reminder, frame, mode) { | |||||||
|     // populate channels |     // populate channels | ||||||
|     set_channels(frame.querySelector("select.channel-selector")); |     set_channels(frame.querySelector("select.channel-selector")); | ||||||
|  |  | ||||||
|     frame.querySelector(`*[name="interval_hours"]`).value = 0; |  | ||||||
|     frame.querySelector(`*[name="interval_minutes"]`).value = 0; |  | ||||||
|  |  | ||||||
|     // populate majority of items |     // populate majority of items | ||||||
|     for (let prop in reminder) { |     for (let prop in reminder) { | ||||||
|         if (reminder.hasOwnProperty(prop) && reminder[prop] !== null) { |         if (reminder.hasOwnProperty(prop) && reminder[prop] !== null) { | ||||||
| @@ -356,8 +351,6 @@ function deserialize_reminder(reminder, frame, mode) { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     update_interval(frame); |  | ||||||
|  |  | ||||||
|     const lastChild = frame.querySelector("div.embed-multifield-box .embed-field-box"); |     const lastChild = frame.querySelector("div.embed-multifield-box .embed-field-box"); | ||||||
|  |  | ||||||
|     for (let field of reminder["embed_fields"]) { |     for (let field of reminder["embed_fields"]) { | ||||||
| @@ -504,8 +497,6 @@ document.addEventListener("remindersLoaded", (event) => { | |||||||
|                 .then((response) => response.json()) |                 .then((response) => response.json()) | ||||||
|                 .then((data) => { |                 .then((data) => { | ||||||
|                     for (let error of data.errors) show_error(error); |                     for (let error of data.errors) show_error(error); | ||||||
|  |  | ||||||
|                     deserialize_reminder(data.reminder, node, "reload"); |  | ||||||
|                 }); |                 }); | ||||||
|  |  | ||||||
|             $saveBtn.querySelector("span.icon > i").classList = ["fas fa-check"]; |             $saveBtn.querySelector("span.icon > i").classList = ["fas fa-check"]; | ||||||
| @@ -724,7 +715,6 @@ $createReminderBtn.addEventListener("click", async () => { | |||||||
|     let reminder = await serialize_reminder($createReminder, "create"); |     let reminder = await serialize_reminder($createReminder, "create"); | ||||||
|     if (reminder.error) { |     if (reminder.error) { | ||||||
|         show_error(reminder.error); |         show_error(reminder.error); | ||||||
|         $createReminderBtn.querySelector("span.icon > i").classList = ["fas fa-sparkles"]; |  | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -842,6 +832,13 @@ $deleteTemplateBtn.addEventListener("click", (ev) => { | |||||||
|         }); |         }); | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  | document.querySelectorAll("textarea.autoresize").forEach((element) => { | ||||||
|  |     element.addEventListener("input", () => { | ||||||
|  |         element.style.height = ""; | ||||||
|  |         element.style.height = element.scrollHeight + 3 + "px"; | ||||||
|  |     }); | ||||||
|  | }); | ||||||
|  |  | ||||||
| let $img; | let $img; | ||||||
| const $urlModal = document.querySelector("div#addImageModal"); | const $urlModal = document.querySelector("div#addImageModal"); | ||||||
| const $urlInput = $urlModal.querySelector("input"); | const $urlInput = $urlModal.querySelector("input"); | ||||||
| @@ -897,13 +894,6 @@ document.addEventListener("remindersLoaded", () => { | |||||||
|                 window.getComputedStyle($discordFrame).borderLeftColor; |                 window.getComputedStyle($discordFrame).borderLeftColor; | ||||||
|         }); |         }); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     document.querySelectorAll("textarea.autoresize").forEach((element) => { |  | ||||||
|         element.addEventListener("input", () => { |  | ||||||
|             element.style.height = ""; |  | ||||||
|             element.style.height = element.scrollHeight + 3 + "px"; |  | ||||||
|         }); |  | ||||||
|     }); |  | ||||||
| }); | }); | ||||||
|  |  | ||||||
| function check_embed_fields() { | function check_embed_fields() { | ||||||
|   | |||||||
| @@ -191,8 +191,19 @@ | |||||||
|                     </label> |                     </label> | ||||||
|                 </div> |                 </div> | ||||||
|             </div> |             </div> | ||||||
|  |             <div class="control"> | ||||||
|  |                 <div class="field"> | ||||||
|  |                     <label> | ||||||
|  |                         <input type="radio" class="default-width" name="exportSelect" value="reminder_templates"> | ||||||
|  |                         Reminder templates | ||||||
|  |                     </label> | ||||||
|  |                 </div> | ||||||
|  |             </div> | ||||||
|             <br> |             <br> | ||||||
|             <div class="has-text-centered"> |             <div class="has-text-centered"> | ||||||
|  |                 <div style="color: red; font-weight: bold;"> | ||||||
|  |                     By selecting "Import", you understand that this will overwrite existing data. | ||||||
|  |                 </div> | ||||||
|                 <div style="color: red"> |                 <div style="color: red"> | ||||||
|                     Please first read the <a href="/help/iemanager">support page</a> |                     Please first read the <a href="/help/iemanager">support page</a> | ||||||
|                 </div> |                 </div> | ||||||
|   | |||||||
| @@ -27,7 +27,7 @@ | |||||||
|             </div> |             </div> | ||||||
|             <div class="tile is-parent"> |             <div class="tile is-parent"> | ||||||
|                 <article class="tile is-child notification"> |                 <article class="tile is-child notification"> | ||||||
|                     <p class="title">Create reminders</p> |                     <p class="title">Creating reminders</p> | ||||||
|                     <p class="subtitle">Learn to create reminders for your server</p> |                     <p class="subtitle">Learn to create reminders for your server</p> | ||||||
|                     <div class="content has-text-centered"> |                     <div class="content has-text-centered"> | ||||||
|                         <a class="button is-size-4 is-rounded is-light" href="/help/create_reminder"> |                         <a class="button is-size-4 is-rounded is-light" href="/help/create_reminder"> | ||||||
| @@ -52,47 +52,47 @@ | |||||||
|                 </article> |                 </article> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
| <!--        <div class="tile is-ancestor">--> |         <div class="tile is-ancestor"> | ||||||
| <!--            <div class="tile is-parent">--> |             <div class="tile is-parent"> | ||||||
| <!--                <article class="tile is-child notification">--> |                 <article class="tile is-child notification"> | ||||||
| <!--                    <p class="title">Timers</p>--> |                     <p class="title">Timers</p> | ||||||
| <!--                    <p class="subtitle">Learn to manage timers</p>--> |                     <p class="subtitle">Learn to manage timers</p> | ||||||
| <!--                    <div class="content has-text-centered">--> |                     <div class="content has-text-centered"> | ||||||
| <!--                        <a class="button is-size-4 is-rounded is-light" href="/help/timers">--> |                         <a class="button is-size-4 is-rounded is-light" href="/help/timers"> | ||||||
| <!--                            <p class="is-size-4">--> |                             <p class="is-size-4"> | ||||||
| <!--                                Read <span class="icon"><i class="fas fa-chevron-right"></i></span>--> |                                 Read <span class="icon"><i class="fas fa-chevron-right"></i></span> | ||||||
| <!--                            </p>--> |                             </p> | ||||||
| <!--                        </a>--> |                         </a> | ||||||
| <!--                    </div>--> |                     </div> | ||||||
| <!--                </article>--> |                 </article> | ||||||
| <!--            </div>--> |             </div> | ||||||
| <!--            <div class="tile is-parent">--> |             <div class="tile is-parent"> | ||||||
| <!--                <article class="tile is-child notification">--> |                 <article class="tile is-child notification"> | ||||||
| <!--                    <p class="title">Todo Lists</p>--> |                     <p class="title">Todo Lists</p> | ||||||
| <!--                    <p class="subtitle">Learn to manage various todo lists</p>--> |                     <p class="subtitle">Learn to manage various todo lists</p> | ||||||
| <!--                    <div class="content has-text-centered">--> |                     <div class="content has-text-centered"> | ||||||
| <!--                        <a class="button is-size-4 is-rounded is-light" href="/help/todo_lists">--> |                         <a class="button is-size-4 is-rounded is-light" href="/help/todo_lists"> | ||||||
| <!--                            <p class="is-size-4">--> |                             <p class="is-size-4"> | ||||||
| <!--                                Read <span class="icon"><i class="fas fa-chevron-right"></i></span>--> |                                 Read <span class="icon"><i class="fas fa-chevron-right"></i></span> | ||||||
| <!--                            </p>--> |                             </p> | ||||||
| <!--                        </a>--> |                         </a> | ||||||
| <!--                    </div>--> |                     </div> | ||||||
| <!--                </article>--> |                 </article> | ||||||
| <!--            </div>--> |             </div> | ||||||
| <!--            <div class="tile is-parent is-vertical">--> |             <div class="tile is-parent is-vertical"> | ||||||
| <!--                <article class="tile is-child notification">--> |                 <article class="tile is-child notification"> | ||||||
| <!--                    <p class="title">Macros</p>--> |                     <p class="title">Macros</p> | ||||||
| <!--                    <p class="subtitle">Learn how to create combination commands called macros, to suit advanced use cases</p>--> |                     <p class="subtitle">Learn how to create combination commands called macros, to suit advanced use cases</p> | ||||||
| <!--                    <div class="content has-text-centered">--> |                     <div class="content has-text-centered"> | ||||||
| <!--                        <a class="button is-size-4 is-rounded is-light" href="/help/macros">--> |                         <a class="button is-size-4 is-rounded is-light" href="/help/macros"> | ||||||
| <!--                            <p class="is-size-4">--> |                             <p class="is-size-4"> | ||||||
| <!--                                Read <span class="icon"><i class="fas fa-chevron-right"></i></span>--> |                                 Read <span class="icon"><i class="fas fa-chevron-right"></i></span> | ||||||
| <!--                            </p>--> |                             </p> | ||||||
| <!--                        </a>--> |                         </a> | ||||||
| <!--                    </div>--> |                     </div> | ||||||
| <!--                </article>--> |                 </article> | ||||||
| <!--            </div>--> |             </div> | ||||||
| <!--        </div>--> |         </div> | ||||||
|         <div class="tile is-ancestor"> |         <div class="tile is-ancestor"> | ||||||
|             <div class="tile is-parent"> |             <div class="tile is-parent"> | ||||||
|                 <article class="tile is-child notification"> |                 <article class="tile is-child notification"> | ||||||
| @@ -107,6 +107,19 @@ | |||||||
|                     </div> |                     </div> | ||||||
|                 </article> |                 </article> | ||||||
|             </div> |             </div> | ||||||
|  |             <div class="tile is-parent"> | ||||||
|  |                 <article class="tile is-child notification"> | ||||||
|  |                     <p class="title">Dashboard</p> | ||||||
|  |                     <p class="subtitle">Learn to use the interactive web dashboard</p> | ||||||
|  |                     <div class="content has-text-centered"> | ||||||
|  |                         <a class="button is-size-4 is-rounded is-light" href="/help/dashboard"> | ||||||
|  |                             <p class="is-size-4"> | ||||||
|  |                                 Read <span class="icon"><i class="fas fa-chevron-right"></i></span> | ||||||
|  |                             </p> | ||||||
|  |                         </a> | ||||||
|  |                     </div> | ||||||
|  |                 </article> | ||||||
|  |             </div> | ||||||
|             <div class="tile is-parent is-vertical"> |             <div class="tile is-parent is-vertical"> | ||||||
|                 <article class="tile is-child notification"> |                 <article class="tile is-child notification"> | ||||||
|                     <p class="title">Import/Export</p> |                     <p class="title">Import/Export</p> | ||||||
| @@ -120,19 +133,6 @@ | |||||||
|                     </div> |                     </div> | ||||||
|                 </article> |                 </article> | ||||||
|             </div> |             </div> | ||||||
|             <div class="tile is-parent"> |  | ||||||
| <!--                <article class="tile is-child notification">--> |  | ||||||
| <!--                    <p class="title">Dashboard</p>--> |  | ||||||
| <!--                    <p class="subtitle">Learn to use the interactive web dashboard</p>--> |  | ||||||
| <!--                    <div class="content has-text-centered">--> |  | ||||||
| <!--                        <a class="button is-size-4 is-rounded is-light" href="/help/dashboard">--> |  | ||||||
| <!--                            <p class="is-size-4">--> |  | ||||||
| <!--                                Read <span class="icon"><i class="fas fa-chevron-right"></i></span>--> |  | ||||||
| <!--                            </p>--> |  | ||||||
| <!--                        </a>--> |  | ||||||
| <!--                    </div>--> |  | ||||||
| <!--                </article>--> |  | ||||||
|             </div> |  | ||||||
|         </div> |         </div> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -28,10 +28,7 @@ | |||||||
|             <div class="container"> |             <div class="container"> | ||||||
|                 <p class="title">Create reminders via the dashboard</p> |                 <p class="title">Create reminders via the dashboard</p> | ||||||
|                 <p class="content"> |                 <p class="content"> | ||||||
|                     Reminders can also be created on the dashboard. The dashboard offers more options for configuring |                     Reminders can also be created on the dashboard. | ||||||
|                     reminders, and offers templates for quick recreation of reminders. |  | ||||||
|  |  | ||||||
|                     <a href="/dashboard">Access the dashboard.</a> |  | ||||||
|                 </p> |                 </p> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
|   | |||||||
| @@ -12,7 +12,7 @@ | |||||||
|     <section class="hero is-small"> |     <section class="hero is-small"> | ||||||
|         <div class="hero-body"> |         <div class="hero-body"> | ||||||
|             <div class="container"> |             <div class="container"> | ||||||
|                 <p class="title">Export data</p> |                 <p class="title">Export your data</p> | ||||||
|                 <p class="content"> |                 <p class="content"> | ||||||
|                     You can export data associated with your server from the dashboard. The data will export as a CSV |                     You can export data associated with your server from the dashboard. The data will export as a CSV | ||||||
|                     file. The CSV file can then be edited and imported to bulk edit server data. |                     file. The CSV file can then be edited and imported to bulk edit server data. | ||||||
| @@ -26,7 +26,8 @@ | |||||||
|             <div class="container"> |             <div class="container"> | ||||||
|                 <p class="title">Import data</p> |                 <p class="title">Import data</p> | ||||||
|                 <p class="content"> |                 <p class="content"> | ||||||
|                     You can import previous exports or modified exports. When importing a file, the new data will be added alongside existing data. |                     You can import previous exports or modified exports. When importing a file, <strong>existing data | ||||||
|  |                     will be overwritten</strong>. | ||||||
|                 </p> |                 </p> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
| @@ -54,7 +55,7 @@ | |||||||
|                         </figure> |                         </figure> | ||||||
|                     </li> |                     </li> | ||||||
|                     <li> |                     <li> | ||||||
|                         Make edits to the spreadsheet. You can add, edit, and remove rows for reminders. Don't remove the top-most (title) row. |                         Make edits to the spreadsheet. You can add, edit, and remove rows for reminders. Don't remove the title row. | ||||||
|                         <figure> |                         <figure> | ||||||
|                             <img src="/static/img/support/iemanager/edit_spreadsheet.png" alt="Editing spreadsheet"> |                             <img src="/static/img/support/iemanager/edit_spreadsheet.png" alt="Editing spreadsheet"> | ||||||
|                         </figure> |                         </figure> | ||||||
| @@ -69,7 +70,7 @@ | |||||||
|                 Other spreadsheet tools can also be used to edit exports, as long as they are properly configured: |                 Other spreadsheet tools can also be used to edit exports, as long as they are properly configured: | ||||||
|                 <ul> |                 <ul> | ||||||
|                     <li> |                     <li> | ||||||
|                         <strong>Google Sheets</strong>: Create a new blank spreadsheet. Select <strong>File > Import > Upload > export.csv</strong>. |                         <strong>Google Sheets</strong>: Create a new blank spreadsheet. Select <strong>File >> Import >> Upload >> export.csv</strong>. | ||||||
|                         Use the following import settings: |                         Use the following import settings: | ||||||
|                         <figure> |                         <figure> | ||||||
|                             <img src="/static/img/support/iemanager/sheets_settings.png" alt="Google sheets import settings"> |                             <img src="/static/img/support/iemanager/sheets_settings.png" alt="Google sheets import settings"> | ||||||
|   | |||||||
| @@ -49,7 +49,7 @@ | |||||||
|                 <p class="content"> |                 <p class="content"> | ||||||
|                     Monthly or yearly intervals are configured the same as fixed intervals. Instead of a fixed time |                     Monthly or yearly intervals are configured the same as fixed intervals. Instead of a fixed time | ||||||
|                     interval, these reminders repeat on a certain day each month or each year. This makes them ideal |                     interval, these reminders repeat on a certain day each month or each year. This makes them ideal | ||||||
|                     for marking calendar events. |                     for marking certain dates. | ||||||
|                 </p> |                 </p> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
| @@ -61,8 +61,7 @@ | |||||||
|                 <p class="title">Interval expiration</p> |                 <p class="title">Interval expiration</p> | ||||||
|                 <p class="content"> |                 <p class="content"> | ||||||
|                     An expiration time can also be specified, both via commands and dashboard, for repeating reminders. |                     An expiration time can also be specified, both via commands and dashboard, for repeating reminders. | ||||||
|                     This is optional, and if omitted, the reminder will repeat indefinitely. Otherwise, the reminder |                     This is optional, and if omitted, the reminder will repeat indefinitely. | ||||||
|                     will be deleted once the expiration date is reached. |  | ||||||
|                 </p> |                 </p> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user