Compare commits
	
		
			27 Commits
		
	
	
		
			jellywx/gu
			...
			4b42966284
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 4b42966284 | |||
| 523ab7f03a | |||
| 6e831c8253 | |||
|  | 4416e5d175 | ||
|  | 734a39a001 | ||
|  | 98191d29ee | ||
|  | 1c4c4a8b31 | ||
|  | d496c81003 | ||
|  | 094d210f64 | ||
|  | 314c72e132 | ||
|  | 4e0163f2cb | ||
|  | e5b8c418af | ||
|  | 3ef8584189 | ||
|  | df2ad09c86 | ||
|  | d70fb24eb1 | ||
|  | 3150c7267d | ||
|  | 6e65e4ff3d | ||
|  | 67a4db2e9a | ||
|  | e9bcb1973f | ||
|  | 9b87fd4258 | ||
|  | a49a849917 | ||
|  | aa74a7f9a3 | ||
|  | 08e4c6cb57 | ||
|  | 6e087bd2dd | ||
| e9792e6322 | |||
| 130504b964 | |||
| 2a8117d0c1 | 
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -2,6 +2,4 @@ | ||||
| .env | ||||
| /venv | ||||
| .cargo | ||||
| assets | ||||
| out.json | ||||
| /.idea | ||||
|   | ||||
							
								
								
									
										1215
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1215
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										34
									
								
								Cargo.toml
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								Cargo.toml
									
									
									
									
									
								
							| @@ -1,20 +1,22 @@ | ||||
| [package] | ||||
| name = "reminder_rs" | ||||
| version = "1.6.6" | ||||
| authors = ["jellywx <judesouthworth@pm.me>"] | ||||
| edition = "2018" | ||||
| name = "reminder-rs" | ||||
| version = "1.6.10" | ||||
| authors = ["Jude Southworth <judesouthworth@pm.me>"] | ||||
| edition = "2021" | ||||
| license = "AGPL-3.0 only" | ||||
| description = "Reminder Bot for Discord, now in Rust" | ||||
|  | ||||
| [dependencies] | ||||
| poise = "0.3" | ||||
| poise = "0.4" | ||||
| dotenv = "0.15" | ||||
| tokio = { version = "1", features = ["process", "full"] } | ||||
| reqwest = "0.11" | ||||
| lazy-regex = "2.3.0" | ||||
| regex = "1.6" | ||||
| log = "0.4" | ||||
| env_logger = "0.9" | ||||
| env_logger = "0.10" | ||||
| chrono = "0.4" | ||||
| chrono-tz = { version = "0.6", features = ["serde"] } | ||||
| chrono-tz = { version = "0.8", features = ["serde"] } | ||||
| lazy_static = "1.4" | ||||
| num-integer = "0.1" | ||||
| serde = "1.0" | ||||
| @@ -23,7 +25,7 @@ serde_repr = "0.1" | ||||
| rmp-serde = "1.1" | ||||
| rand = "0.8" | ||||
| levenshtein = "1.0" | ||||
| sqlx = { version = "0.6", features = ["runtime-tokio-rustls", "macros", "mysql", "bigdecimal", "chrono"]} | ||||
| sqlx = { version = "0.6", features = ["runtime-tokio-rustls", "macros", "mysql", "bigdecimal", "chrono", "migrate"]} | ||||
| base64 = "0.13" | ||||
|  | ||||
| [dependencies.postman] | ||||
| @@ -31,3 +33,19 @@ path = "postman" | ||||
|  | ||||
| [dependencies.reminder_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 | ||||
|   | ||||
							
								
								
									
										9
									
								
								Containerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								Containerfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| FROM ubuntu:20.04 | ||||
|  | ||||
| ENV RUSTUP_HOME=/usr/local/rustup \ | ||||
|     CARGO_HOME=/usr/local/cargo \ | ||||
|     PATH=/usr/local/cargo/bin:$PATH | ||||
|  | ||||
| RUN apt update && DEBIAN_FRONTEND=noninteractive TZ=Etc/UTC apt install -y gcc gcc-multilib cmake pkg-config libssl-dev curl mysql-client-8.0 | ||||
| RUN curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y --no-modify-path --profile minimal --default-toolchain nightly | ||||
| RUN cargo install cargo-deb | ||||
							
								
								
									
										42
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										42
									
								
								README.md
									
									
									
									
									
								
							| @@ -7,25 +7,30 @@ 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) | ||||
|  | ||||
| ### Compiling | ||||
| Install build requirements:  | ||||
| `sudo apt install gcc gcc-multilib cmake libssl-dev build-essential` | ||||
| ### Compiling for local target | ||||
| 1. Install requirements:  | ||||
| `sudo apt install gcc gcc-multilib cmake libssl-dev build-essential python3-dateparser` | ||||
| 2. Install rustup from https://rustup.rs | ||||
| 3. Install the nightly toolchain: `rustup toolchain default nightly` | ||||
| 4. Install database server: `sudo apt install mysql-server-8.0`. Create a database called `reminders`. | ||||
| 5. Install `sqlx-cli`: `cargo install sqlx-cli`. | ||||
| 6. Run migrations: `sqlx migrate run`. | ||||
| 7. Set environment variables: | ||||
|    * `DATABASE_URL` - the URL of your MySQL database (`mysql://user[:password]@domain/database`) | ||||
| 8. Build: `cargo build --release` | ||||
|  | ||||
| Install Rust from https://rustup.rs | ||||
| ### Compiling for other target | ||||
| By default, this builds targeting Ubuntu 20.04. Modify the Containerfile if you wish to target a different platform. These instructions are written using `podman`, but `docker` should work too. | ||||
|  | ||||
| Reminder Bot can then be built by running `cargo build --release` in the top level directory. It is necessary to create a  | ||||
| folder called 'assets' containing an image file with its name specified in the environment as `WEBHOOK_AVATAR`, of  | ||||
| dimensions 128x128px to be used as the webhook avatar. | ||||
| 1. Install container software: `sudo apt install podman`. | ||||
| 2. Install database server: `sudo apt install mysql-server-8.0`. Create a database called `reminders` | ||||
| 3. Install SQLx CLI: `cargo install sqlx-cli` | ||||
| 4. From the source code directory, execute `sqlx migrate run` | ||||
| 5. Build container image: `podman build -t reminder-rs .` | ||||
| 6. Build with podman: `podman run --rm --network=host -v "$PWD":/mnt -w /mnt -e "DATABASE_URL=mysql://user@localhost/reminders" reminder-rs cargo deb`  | ||||
|  | ||||
| #### 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** | ||||
|  | ||||
| ### 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 | ||||
| ### Configuring | ||||
| 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__ | ||||
| @@ -37,10 +42,5 @@ __Other Variables__ | ||||
| * `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 | ||||
| * `CNC_GUILD` - default `None`, accepts a single Discord guild ID for the server that the subscription roles belong to | ||||
| * `PYTHON_LOCATION` - default `venv/bin/python3`. Can be changed if your Python executable is located somewhere else | ||||
| * `PYTHON_LOCATION` - default `/usr/bin/python3`. Can be changed if your Python executable is located somewhere else | ||||
| * `THEME_COLOR` - default `8fb677`. Specifies the hex value of the color to use on info message embeds  | ||||
| * `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] | ||||
| address = "0.0.0.0" | ||||
| port = 5000 | ||||
| port = 18920 | ||||
| template_dir = "web/templates" | ||||
| limits = { json = "10MiB" } | ||||
|  | ||||
| @@ -11,18 +11,18 @@ secret_key = "tR8krio5FXTnnyIZNiJDXPondz0kI1v6X6BXZcBGIRY=" | ||||
| certs = "web/private/rsa_sha256_cert.pem" | ||||
| key = "web/private/rsa_sha256_key.pem" | ||||
|  | ||||
| [rsa_sha256.tls] | ||||
| [debug.rsa_sha256.tls] | ||||
| certs = "web/private/rsa_sha256_cert.pem" | ||||
| key = "web/private/rsa_sha256_key.pem" | ||||
|  | ||||
| [ecdsa_nistp256_sha256.tls] | ||||
| [debug.ecdsa_nistp256_sha256.tls] | ||||
| certs = "web/private/ecdsa_nistp256_sha256_cert.pem" | ||||
| key = "web/private/ecdsa_nistp256_sha256_key_pkcs8.pem" | ||||
|  | ||||
| [ecdsa_nistp384_sha384.tls] | ||||
| [debug.ecdsa_nistp384_sha384.tls] | ||||
| certs = "web/private/ecdsa_nistp384_sha384_cert.pem" | ||||
| key = "web/private/ecdsa_nistp384_sha384_key_pkcs8.pem" | ||||
|  | ||||
| [ed25519.tls] | ||||
| [debug.ed25519.tls] | ||||
| certs = "web/private/ed25519_cert.pem" | ||||
| key = "eb/private/ed25519_key.pem" | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								assets/webhook.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								assets/webhook.jpg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 21 KiB | 
							
								
								
									
										3
									
								
								build.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								build.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| fn main() { | ||||
|     println!("cargo:rerun-if-changed=migrations"); | ||||
| } | ||||
							
								
								
									
										16
									
								
								conf/default.env
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								conf/default.env
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| 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
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								debian/postinst
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| #!/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
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								debian/postrm
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| #!/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,4 +0,0 @@ | ||||
| USE reminders; | ||||
|  | ||||
| ALTER TABLE reminders.reminders RENAME COLUMN `interval` TO `interval_seconds`; | ||||
| ALTER TABLE reminders.reminders ADD COLUMN `interval_months` INT UNSIGNED DEFAULT NULL; | ||||
| @@ -1,10 +1,6 @@ | ||||
| CREATE DATABASE IF NOT EXISTS reminders; | ||||
| 
 | ||||
| SET FOREIGN_KEY_CHECKS=0; | ||||
| 
 | ||||
| USE reminders; | ||||
| 
 | ||||
| CREATE TABLE reminders.guilds ( | ||||
| CREATE TABLE guilds ( | ||||
|     id INT UNSIGNED UNIQUE NOT NULL AUTO_INCREMENT, | ||||
|     guild BIGINT UNSIGNED UNIQUE NOT NULL, | ||||
| 
 | ||||
| @@ -18,10 +14,10 @@ CREATE TABLE reminders.guilds ( | ||||
|     default_avatar VARCHAR(512) DEFAULT 'https://raw.githubusercontent.com/reminder-bot/logos/master/Remind_Me_Bot_Logo_PPic.jpg' NOT NULL, | ||||
| 
 | ||||
|     PRIMARY KEY (id), | ||||
|     FOREIGN KEY (default_channel_id) REFERENCES reminders.channels(id) ON DELETE SET NULL | ||||
|     FOREIGN KEY (default_channel_id) REFERENCES channels(id) ON DELETE SET NULL | ||||
| ); | ||||
| 
 | ||||
| CREATE TABLE reminders.channels ( | ||||
| CREATE TABLE channels ( | ||||
|     id INT UNSIGNED UNIQUE NOT NULL AUTO_INCREMENT, | ||||
|     channel BIGINT UNSIGNED UNIQUE NOT NULL, | ||||
| 
 | ||||
| @@ -39,10 +35,10 @@ CREATE TABLE reminders.channels ( | ||||
|     guild_id INT UNSIGNED, | ||||
| 
 | ||||
|     PRIMARY KEY (id), | ||||
|     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE | ||||
|     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE | ||||
| ); | ||||
| 
 | ||||
| CREATE TABLE reminders.users ( | ||||
| CREATE TABLE users ( | ||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||
|     user BIGINT UNSIGNED UNIQUE NOT NULL, | ||||
| 
 | ||||
| @@ -59,10 +55,10 @@ CREATE TABLE reminders.users ( | ||||
|     patreon BOOLEAN NOT NULL DEFAULT 0, | ||||
| 
 | ||||
|     PRIMARY KEY (id), | ||||
|     FOREIGN KEY (dm_channel) REFERENCES reminders.channels(id) ON DELETE RESTRICT | ||||
|     FOREIGN KEY (dm_channel) REFERENCES channels(id) ON DELETE RESTRICT | ||||
| ); | ||||
| 
 | ||||
| CREATE TABLE reminders.roles ( | ||||
| CREATE TABLE roles ( | ||||
|     id INT UNSIGNED UNIQUE NOT NULL AUTO_INCREMENT, | ||||
|     role BIGINT UNSIGNED UNIQUE NOT NULL, | ||||
| 
 | ||||
| @@ -71,10 +67,10 @@ CREATE TABLE reminders.roles ( | ||||
|     guild_id INT UNSIGNED NOT NULL, | ||||
| 
 | ||||
|     PRIMARY KEY (id), | ||||
|     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE | ||||
|     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE | ||||
| ); | ||||
| 
 | ||||
| CREATE TABLE reminders.embeds ( | ||||
| CREATE TABLE embeds ( | ||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||
| 
 | ||||
|     title VARCHAR(256) NOT NULL DEFAULT '', | ||||
| @@ -91,7 +87,7 @@ CREATE TABLE reminders.embeds ( | ||||
|     PRIMARY KEY (id) | ||||
| ); | ||||
| 
 | ||||
| CREATE TABLE reminders.embed_fields ( | ||||
| CREATE TABLE embed_fields ( | ||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||
| 
 | ||||
|     title VARCHAR(256) NOT NULL DEFAULT '', | ||||
| @@ -100,10 +96,10 @@ CREATE TABLE reminders.embed_fields ( | ||||
|     embed_id INT UNSIGNED NOT NULL, | ||||
| 
 | ||||
|     PRIMARY KEY (id), | ||||
|     FOREIGN KEY (embed_id) REFERENCES reminders.embeds(id) ON DELETE CASCADE | ||||
|     FOREIGN KEY (embed_id) REFERENCES embeds(id) ON DELETE CASCADE | ||||
| ); | ||||
| 
 | ||||
| CREATE TABLE reminders.messages ( | ||||
| CREATE TABLE messages ( | ||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||
| 
 | ||||
|     content VARCHAR(2048) NOT NULL DEFAULT '', | ||||
| @@ -114,10 +110,10 @@ CREATE TABLE reminders.messages ( | ||||
|     attachment_name VARCHAR(260), | ||||
| 
 | ||||
|     PRIMARY KEY (id), | ||||
|     FOREIGN KEY (embed_id) REFERENCES reminders.embeds(id) ON DELETE SET NULL | ||||
|     FOREIGN KEY (embed_id) REFERENCES embeds(id) ON DELETE SET NULL | ||||
| ); | ||||
| 
 | ||||
| CREATE TABLE reminders.reminders ( | ||||
| CREATE TABLE reminders ( | ||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||
|     uid VARCHAR(64) UNIQUE NOT NULL, | ||||
| 
 | ||||
| @@ -140,20 +136,20 @@ CREATE TABLE reminders.reminders ( | ||||
|     set_by INT UNSIGNED, | ||||
| 
 | ||||
|     PRIMARY KEY (id), | ||||
|     FOREIGN KEY (message_id) REFERENCES reminders.messages(id) ON DELETE RESTRICT, | ||||
|     FOREIGN KEY (channel_id) REFERENCES reminders.channels(id) ON DELETE CASCADE, | ||||
|     FOREIGN KEY (set_by) REFERENCES reminders.users(id) ON DELETE SET NULL | ||||
|     FOREIGN KEY (message_id) REFERENCES messages(id) ON DELETE RESTRICT, | ||||
|     FOREIGN KEY (channel_id) REFERENCES channels(id) ON DELETE CASCADE, | ||||
|     FOREIGN KEY (set_by) REFERENCES users(id) ON DELETE SET NULL | ||||
| ); | ||||
| 
 | ||||
| CREATE TRIGGER message_cleanup AFTER DELETE ON reminders.reminders | ||||
| CREATE TRIGGER message_cleanup AFTER DELETE ON reminders | ||||
| FOR EACH ROW | ||||
|     DELETE FROM reminders.messages WHERE id = OLD.message_id; | ||||
|     DELETE FROM messages WHERE id = OLD.message_id; | ||||
| 
 | ||||
| CREATE TRIGGER embed_cleanup AFTER DELETE ON reminders.messages | ||||
| CREATE TRIGGER embed_cleanup AFTER DELETE ON messages | ||||
| FOR EACH ROW | ||||
|     DELETE FROM reminders.embeds WHERE id = OLD.embed_id; | ||||
|     DELETE FROM embeds WHERE id = OLD.embed_id; | ||||
| 
 | ||||
| CREATE TABLE reminders.todos ( | ||||
| CREATE TABLE todos ( | ||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||
|     user_id INT UNSIGNED, | ||||
|     guild_id INT UNSIGNED, | ||||
| @@ -161,23 +157,23 @@ CREATE TABLE reminders.todos ( | ||||
|     value VARCHAR(2000) NOT NULL, | ||||
| 
 | ||||
|     PRIMARY KEY (id), | ||||
|     FOREIGN KEY (user_id) REFERENCES reminders.users(id) ON DELETE SET NULL, | ||||
|     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE, | ||||
|     FOREIGN KEY (channel_id) REFERENCES reminders.channels(id) ON DELETE SET NULL | ||||
|     FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE SET NULL, | ||||
|     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE, | ||||
|     FOREIGN KEY (channel_id) REFERENCES channels(id) ON DELETE SET NULL | ||||
| ); | ||||
| 
 | ||||
| CREATE TABLE reminders.command_restrictions ( | ||||
| CREATE TABLE command_restrictions ( | ||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||
| 
 | ||||
|     role_id INT UNSIGNED NOT NULL, | ||||
|     command ENUM('todos', 'natural', 'remind', 'interval', 'timer', 'del', 'look', 'alias', 'countdown') NOT NULL, | ||||
| 
 | ||||
|     PRIMARY KEY (id), | ||||
|     FOREIGN KEY (role_id) REFERENCES reminders.roles(id) ON DELETE CASCADE, | ||||
|     FOREIGN KEY (role_id) REFERENCES roles(id) ON DELETE CASCADE, | ||||
|     UNIQUE KEY (`role_id`, `command`) | ||||
| ); | ||||
| 
 | ||||
| CREATE TABLE reminders.timers ( | ||||
| CREATE TABLE timers ( | ||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||
|     start_time TIMESTAMP NOT NULL DEFAULT NOW(), | ||||
|     name VARCHAR(32) NOT NULL, | ||||
| @@ -186,7 +182,7 @@ CREATE TABLE reminders.timers ( | ||||
|     PRIMARY KEY (id) | ||||
| ); | ||||
| 
 | ||||
| CREATE TABLE reminders.events ( | ||||
| CREATE TABLE events ( | ||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||
|     `time` TIMESTAMP NOT NULL DEFAULT NOW(), | ||||
| 
 | ||||
| @@ -198,12 +194,12 @@ CREATE TABLE reminders.events ( | ||||
|     reminder_id INT UNSIGNED, | ||||
| 
 | ||||
|     PRIMARY KEY (id), | ||||
|     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE, | ||||
|     FOREIGN KEY (user_id) REFERENCES reminders.users(id) ON DELETE SET NULL, | ||||
|     FOREIGN KEY (reminder_id) REFERENCES reminders.reminders(id) ON DELETE SET NULL | ||||
|     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE, | ||||
|     FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE SET NULL, | ||||
|     FOREIGN KEY (reminder_id) REFERENCES reminders(id) ON DELETE SET NULL | ||||
| ); | ||||
| 
 | ||||
| CREATE TABLE reminders.command_aliases ( | ||||
| CREATE TABLE command_aliases ( | ||||
|     id INT UNSIGNED AUTO_INCREMENT UNIQUE NOT NULL, | ||||
| 
 | ||||
|     guild_id INT UNSIGNED NOT NULL, | ||||
| @@ -212,22 +208,22 @@ CREATE TABLE reminders.command_aliases ( | ||||
|     command VARCHAR(2048) NOT NULL, | ||||
| 
 | ||||
|     PRIMARY KEY (id), | ||||
|     FOREIGN KEY (guild_id) REFERENCES reminders.guilds(id) ON DELETE CASCADE, | ||||
|     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE, | ||||
|     UNIQUE KEY (`guild_id`, `name`) | ||||
| ); | ||||
| 
 | ||||
| CREATE TABLE reminders.guild_users ( | ||||
| CREATE TABLE guild_users ( | ||||
|     guild INT UNSIGNED NOT NULL, | ||||
|     user INT UNSIGNED NOT NULL, | ||||
| 
 | ||||
|     can_access BOOL NOT NULL DEFAULT 0, | ||||
| 
 | ||||
|     FOREIGN KEY (guild) REFERENCES reminders.guilds(id) ON DELETE CASCADE, | ||||
|     FOREIGN KEY (user) REFERENCES reminders.users(id) ON DELETE CASCADE, | ||||
|     FOREIGN KEY (guild) REFERENCES guilds(id) ON DELETE CASCADE, | ||||
|     FOREIGN KEY (user) REFERENCES users(id) ON DELETE CASCADE, | ||||
|     UNIQUE KEY (guild, user) | ||||
| ); | ||||
| 
 | ||||
| CREATE EVENT reminders.event_cleanup | ||||
| CREATE EVENT event_cleanup | ||||
| ON SCHEDULE AT CURRENT_TIMESTAMP + INTERVAL 1 DAY | ||||
| ON COMPLETION PRESERVE | ||||
| DO DELETE FROM reminders.events WHERE `time` < DATE_SUB(NOW(), INTERVAL 5 DAY); | ||||
| DO DELETE FROM events WHERE `time` < DATE_SUB(NOW(), INTERVAL 5 DAY); | ||||
| @@ -1,5 +1,3 @@ | ||||
| USE reminders; | ||||
| 
 | ||||
| SET FOREIGN_KEY_CHECKS = 0; | ||||
| 
 | ||||
| DROP TABLE IF EXISTS reminders_new; | ||||
| @@ -1,5 +1,3 @@ | ||||
| USE reminders; | ||||
| 
 | ||||
| CREATE TABLE macro ( | ||||
|     id INT UNSIGNED AUTO_INCREMENT, | ||||
|     guild_id INT UNSIGNED NOT NULL, | ||||
| @@ -0,0 +1,2 @@ | ||||
| ALTER TABLE reminders RENAME COLUMN `interval` TO `interval_seconds`; | ||||
| ALTER TABLE reminders ADD COLUMN `interval_months` INT UNSIGNED DEFAULT NULL; | ||||
| @@ -1,5 +1,3 @@ | ||||
| USE reminders; | ||||
| 
 | ||||
| CREATE TABLE reminder_template ( | ||||
|     `id` INT UNSIGNED NOT NULL AUTO_INCREMENT, | ||||
| 
 | ||||
							
								
								
									
										1
									
								
								migrations/20221210000000_reminder_daily_intervals.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								migrations/20221210000000_reminder_daily_intervals.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| ALTER TABLE reminders ADD COLUMN `interval_days` INT UNSIGNED DEFAULT NULL; | ||||
							
								
								
									
										2
									
								
								migrations/20230511125236_reminder_threads.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								migrations/20230511125236_reminder_threads.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| -- Add migration script here | ||||
| ALTER TABLE reminders ADD COLUMN `thread_id` BIGINT DEFAULT NULL; | ||||
							
								
								
									
										41
									
								
								nginx/reminder-rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								nginx/reminder-rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| server { | ||||
|         server_name www.reminder-bot.com; | ||||
|  | ||||
|         return 301 $scheme://reminder-bot.com$request_uri; | ||||
| } | ||||
|  | ||||
| server { | ||||
|         listen 80; | ||||
|         server_name reminder-bot.com; | ||||
|  | ||||
| 	    return 301 https://reminder-bot.com$request_uri; | ||||
| } | ||||
|  | ||||
| server { | ||||
|         listen 443 ssl; | ||||
|         server_name reminder-bot.com; | ||||
|  | ||||
|         ssl_certificate /etc/letsencrypt/live/reminder-bot.com/fullchain.pem; | ||||
|         ssl_certificate_key /etc/letsencrypt/live/reminder-bot.com/privkey.pem; | ||||
|  | ||||
|         access_log /var/log/nginx/access.log; | ||||
|         error_log /var/log/nginx/error.log; | ||||
|  | ||||
|         proxy_buffer_size 128k; | ||||
|         proxy_buffers 4 256k; | ||||
|         proxy_busy_buffers_size 256k; | ||||
|  | ||||
|         location / { | ||||
|                 proxy_pass http://localhost:18920; | ||||
|                 proxy_redirect off; | ||||
|                 proxy_set_header Host $host; | ||||
|                 proxy_set_header X-Real-IP $remote_addr; | ||||
|                 proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; | ||||
| 		        proxy_set_header X-Forwarded-Proto $scheme; | ||||
|         } | ||||
|  | ||||
|         location /static { | ||||
|                 alias /var/www/reminder-rs/static; | ||||
|                 expires 30d; | ||||
|         } | ||||
| } | ||||
| @@ -1,4 +1,4 @@ | ||||
| use chrono::Duration; | ||||
| use chrono::{DateTime, Days, Duration, Months}; | ||||
| use chrono_tz::Tz; | ||||
| use lazy_static::lazy_static; | ||||
| use log::{error, info, warn}; | ||||
| @@ -62,7 +62,8 @@ pub fn substitute(string: &str) -> String { | ||||
|         let format = caps.name("format").map(|m| m.as_str()); | ||||
|  | ||||
|         if let (Some(final_time), Some(format)) = (final_time, format) { | ||||
|             let dt = NaiveDateTime::from_timestamp(final_time, 0); | ||||
|             match NaiveDateTime::from_timestamp_opt(final_time, 0) { | ||||
|                 Some(dt) => { | ||||
|                     let now = Utc::now().naive_utc(); | ||||
|  | ||||
|                     let difference = { | ||||
| @@ -74,6 +75,10 @@ pub fn substitute(string: &str) -> String { | ||||
|                     }; | ||||
|  | ||||
|                     fmt_displacement(format, difference.num_seconds() as u64) | ||||
|                 } | ||||
|  | ||||
|                 None => String::new(), | ||||
|             } | ||||
|         } else { | ||||
|             String::new() | ||||
|         } | ||||
| @@ -243,11 +248,12 @@ pub struct Reminder { | ||||
|     attachment: Option<Vec<u8>>, | ||||
|     attachment_name: Option<String>, | ||||
|  | ||||
|     utc_time: NaiveDateTime, | ||||
|     utc_time: DateTime<Utc>, | ||||
|     timezone: String, | ||||
|     restartable: bool, | ||||
|     expires: Option<NaiveDateTime>, | ||||
|     expires: Option<DateTime<Utc>>, | ||||
|     interval_seconds: Option<u32>, | ||||
|     interval_days: Option<u32>, | ||||
|     interval_months: Option<u32>, | ||||
|  | ||||
|     avatar: Option<String>, | ||||
| @@ -281,6 +287,7 @@ SELECT | ||||
|     reminders.`restartable` AS restartable, | ||||
|     reminders.`expires` AS 'expires', | ||||
|     reminders.`interval_seconds` AS 'interval_seconds', | ||||
|     reminders.`interval_days` AS 'interval_days', | ||||
|     reminders.`interval_months` AS 'interval_months', | ||||
|  | ||||
|     reminders.`avatar` AS avatar, | ||||
| @@ -330,9 +337,7 @@ WHERE | ||||
|  | ||||
|     async fn reset_webhook(&self, pool: impl Executor<'_, Database = Database> + Copy) { | ||||
|         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 | ||||
|         ) | ||||
|         .execute(pool) | ||||
| @@ -341,55 +346,43 @@ UPDATE channels SET webhook_id = NULL, webhook_token = NULL WHERE channel = ? | ||||
|  | ||||
|     async fn refresh(&self, pool: impl Executor<'_, Database = Database> + Copy) { | ||||
|         if self.interval_seconds.is_some() || self.interval_months.is_some() { | ||||
|             let now = Utc::now().naive_local(); | ||||
|             let mut updated_reminder_time = self.utc_time; | ||||
|             let now = Utc::now(); | ||||
|             let mut updated_reminder_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 { | ||||
|                 match sqlx::query!( | ||||
|                     // use the second date_add to force return value to datetime | ||||
|                     "SELECT DATE_ADD(DATE_ADD(?, INTERVAL ? MONTH), INTERVAL 0 SECOND) AS new_time", | ||||
|                     updated_reminder_time, | ||||
|                     interval | ||||
|                 ) | ||||
|                 .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"); | ||||
|                     updated_reminder_time = updated_reminder_time | ||||
|                         .checked_add_months(Months::new(interval)) | ||||
|                         .unwrap_or_else(|| { | ||||
|                             warn!("Could not add months to a reminder"); | ||||
|  | ||||
|                             updated_reminder_time += Duration::days(30); | ||||
|                             updated_reminder_time | ||||
|                         }); | ||||
|                 } | ||||
|                     }, | ||||
|  | ||||
|                     Err(e) => { | ||||
|                         warn!("Could not update interval by months: {:?}", e); | ||||
|                 if let Some(interval) = self.interval_days { | ||||
|                     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"); | ||||
|  | ||||
|                         // naively fallback to adding 30 days | ||||
|                         updated_reminder_time += Duration::days(30); | ||||
|                     } | ||||
|                 } | ||||
|                             updated_reminder_time | ||||
|                         }); | ||||
|                 } | ||||
|  | ||||
|                 if let Some(interval) = self.interval_seconds { | ||||
|                 while updated_reminder_time < now { | ||||
|                     updated_reminder_time += Duration::seconds(interval as i64); | ||||
|                     updated_reminder_time = | ||||
|                         updated_reminder_time + Duration::seconds(interval as i64); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if self.expires.map_or(false, |expires| { | ||||
|                 NaiveDateTime::from_timestamp(updated_reminder_time.timestamp(), 0) > expires | ||||
|             }) { | ||||
|             if self.expires.map_or(false, |expires| updated_reminder_time > expires) { | ||||
|                 self.force_delete(pool).await; | ||||
|             } else { | ||||
|                 sqlx::query!( | ||||
|                     " | ||||
| UPDATE reminders SET `utc_time` = ? WHERE `id` = ? | ||||
|                     ", | ||||
|                     updated_reminder_time, | ||||
|                     "UPDATE reminders SET `utc_time` = ? WHERE `id` = ?", | ||||
|                     updated_reminder_time.with_timezone(&Utc), | ||||
|                     self.id | ||||
|                 ) | ||||
|                 .execute(pool) | ||||
| @@ -402,12 +395,7 @@ UPDATE reminders SET `utc_time` = ? WHERE `id` = ? | ||||
|     } | ||||
|  | ||||
|     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) | ||||
|             .await | ||||
|             .expect(&format!("Could not delete Reminder {}", self.id)); | ||||
| @@ -504,8 +492,10 @@ DELETE FROM reminders WHERE `id` = ? | ||||
|                     w.content(&reminder.content).tts(reminder.tts); | ||||
|  | ||||
|                     if let Some(username) = &reminder.username { | ||||
|                         if !username.is_empty() { | ||||
|                             w.username(username); | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     if let Some(avatar) = &reminder.avatar { | ||||
|                         w.avatar_url(avatar); | ||||
| @@ -548,9 +538,7 @@ DELETE FROM reminders WHERE `id` = ? | ||||
|                     .map_or(true, |inner| inner >= Utc::now().naive_local())) | ||||
|         { | ||||
|             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 | ||||
|             ) | ||||
|             .execute(pool) | ||||
|   | ||||
| @@ -37,20 +37,6 @@ WHERE | ||||
|     .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( | ||||
|     ctx: Context<'_>, | ||||
|     partial: &str, | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| use super::super::autocomplete::macro_name_autocomplete; | ||||
| use crate::{models::command_macro::guild_command_macro, Context, Data, Error}; | ||||
| use crate::{models::command_macro::guild_command_macro, Context, Data, Error, THEME_COLOR}; | ||||
|  | ||||
| /// Run a recorded macro | ||||
| #[poise::command( | ||||
| @@ -17,7 +17,17 @@ pub async fn run_macro( | ||||
| ) -> Result<(), Error> { | ||||
|     match guild_command_macro(&Context::Application(ctx), &name).await { | ||||
|         Some(command_macro) => { | ||||
|             ctx.defer_response(false).await?; | ||||
|             Context::Application(ctx) | ||||
|                 .send(|b| { | ||||
|                     b.embed(|e| { | ||||
|                         e.title("Running Macro").color(*THEME_COLOR).description(format!( | ||||
|                             "Running macro {} ({} commands)", | ||||
|                             command_macro.name, | ||||
|                             command_macro.commands.len() | ||||
|                         )) | ||||
|                     }) | ||||
|                 }) | ||||
|                 .await?; | ||||
|  | ||||
|             for command in command_macro.commands { | ||||
|                 if let Some(action) = command.action { | ||||
|   | ||||
| @@ -1,10 +1,6 @@ | ||||
| use std::{ | ||||
|     collections::HashSet, | ||||
|     string::ToString, | ||||
|     time::{SystemTime, UNIX_EPOCH}, | ||||
| }; | ||||
| use std::{collections::HashSet, string::ToString}; | ||||
|  | ||||
| use chrono::NaiveDateTime; | ||||
| use chrono::{DateTime, NaiveDateTime, Utc}; | ||||
| use chrono_tz::Tz; | ||||
| use num_integer::Integer; | ||||
| use poise::{ | ||||
| @@ -15,9 +11,7 @@ use poise::{ | ||||
| }; | ||||
|  | ||||
| use crate::{ | ||||
|     commands::autocomplete::{ | ||||
|         multiline_autocomplete, time_hint_autocomplete, timezone_autocomplete, | ||||
|     }, | ||||
|     commands::autocomplete::{time_hint_autocomplete, timezone_autocomplete}, | ||||
|     component_models::{ | ||||
|         pager::{DelPager, LookPager, Pager}, | ||||
|         ComponentDataModel, DelSelector, UndoReminder, | ||||
| @@ -62,8 +56,8 @@ pub async fn pause( | ||||
|             let parsed = natural_parser(&until, &timezone.to_string()).await; | ||||
|  | ||||
|             if let Some(timestamp) = parsed { | ||||
|                 let dt = NaiveDateTime::from_timestamp(timestamp, 0); | ||||
|  | ||||
|                 match NaiveDateTime::from_timestamp_opt(timestamp, 0) { | ||||
|                     Some(dt) => { | ||||
|                         channel.paused = true; | ||||
|                         channel.paused_until = Some(dt); | ||||
|  | ||||
| @@ -74,6 +68,15 @@ pub async fn pause( | ||||
|                             timestamp | ||||
|                         )) | ||||
|                         .await?; | ||||
|                     } | ||||
|  | ||||
|                     None => { | ||||
|                         ctx.say(format!( | ||||
|                             "Time processed could not be interpreted as `DateTime`. Please write the time as clearly as possible", | ||||
|                         )) | ||||
|                         .await?; | ||||
|                     } | ||||
|                 } | ||||
|             } else { | ||||
|                 ctx.say( | ||||
|                     "Time could not be processed. Please write the time as clearly as possible", | ||||
| @@ -247,7 +250,7 @@ pub async fn look( | ||||
|                 char_count < EMBED_DESCRIPTION_MAX_LENGTH | ||||
|             }) | ||||
|             .collect::<Vec<String>>() | ||||
|             .join("\n"); | ||||
|             .join(""); | ||||
|  | ||||
|         let pages = reminders | ||||
|             .iter() | ||||
| @@ -434,11 +437,8 @@ pub fn show_delete_page(reminders: &[Reminder], page: usize, timezone: Tz) -> Cr | ||||
|     reply | ||||
| } | ||||
|  | ||||
| fn time_difference(start_time: NaiveDateTime) -> String { | ||||
|     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(); | ||||
| fn time_difference(start_time: DateTime<Utc>) -> String { | ||||
|     let delta = (Utc::now() - start_time).num_seconds(); | ||||
|  | ||||
|     let (minutes, seconds) = delta.div_rem(&60); | ||||
|     let (hours, minutes) = minutes.div_rem(&60); | ||||
| @@ -562,20 +562,17 @@ struct ContentModal { | ||||
|     content: String, | ||||
| } | ||||
|  | ||||
| /// Create a reminder. Press "+4 more" for other options. | ||||
| /// Create a reminder with multi-line content. Press "+4 more" for other options. | ||||
| #[poise::command( | ||||
|     slash_command, | ||||
|     identifying_name = "remind", | ||||
|     identifying_name = "multiline", | ||||
|     default_member_permissions = "MANAGE_GUILD" | ||||
| )] | ||||
| pub async fn remind( | ||||
| pub async fn multiline( | ||||
|     ctx: ApplicationContext<'_>, | ||||
|     #[description = "A description of the time to set the reminder for"] | ||||
|     #[autocomplete = "time_hint_autocomplete"] | ||||
|     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 = "(Patreon only) Time to wait before repeating the reminder. Leave blank for one-shot reminder"] | ||||
|     interval: Option<String>, | ||||
| @@ -588,8 +585,6 @@ pub async fn remind( | ||||
|     timezone: Option<String>, | ||||
| ) -> Result<(), Error> { | ||||
|     let tz = timezone.map(|t| t.parse::<Tz>().ok()).flatten(); | ||||
|  | ||||
|     if content.is_empty() { | ||||
|     let data = ContentModal::execute(ctx).await?; | ||||
|  | ||||
|     create_reminder( | ||||
| @@ -603,19 +598,35 @@ pub async fn remind( | ||||
|         tz, | ||||
|     ) | ||||
|     .await | ||||
|     } else { | ||||
|         create_reminder( | ||||
|             Context::Application(ctx), | ||||
|             time, | ||||
|             content, | ||||
|             channels, | ||||
|             interval, | ||||
|             expires, | ||||
|             tts, | ||||
|             tz, | ||||
|         ) | ||||
| } | ||||
|  | ||||
| /// Create a reminder. Press "+4 more" for other options. Use "/multiline" for multiline content. | ||||
| #[poise::command( | ||||
|     slash_command, | ||||
|     identifying_name = "remind", | ||||
|     default_member_permissions = "MANAGE_GUILD" | ||||
| )] | ||||
| pub async fn remind( | ||||
|     ctx: ApplicationContext<'_>, | ||||
|     #[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 | ||||
|     } | ||||
| } | ||||
|  | ||||
| async fn create_reminder( | ||||
|   | ||||
| @@ -340,7 +340,18 @@ pub fn show_todo_page( | ||||
|                                 opt.create_option(|o| { | ||||
|                                     o.label(format!("Mark {} complete", count + first_num)) | ||||
|                                         .value(id) | ||||
|                                         .description(disp.split_once(' ').unwrap_or(("", "")).1) | ||||
|                                         .description({ | ||||
|                                             let c = disp.split_once(' ').unwrap_or(("", "")).1; | ||||
|  | ||||
|                                             if c.len() > 100 { | ||||
|                                                 format!( | ||||
|                                                     "{}...", | ||||
|                                                     c.chars().take(97).collect::<String>() | ||||
|                                                 ) | ||||
|                                             } else { | ||||
|                                                 c.to_string() | ||||
|                                             } | ||||
|                                         }) | ||||
|                                 }); | ||||
|                             } | ||||
|  | ||||
|   | ||||
| @@ -113,7 +113,7 @@ impl ComponentDataModel { | ||||
|                         char_count < EMBED_DESCRIPTION_MAX_LENGTH | ||||
|                     }) | ||||
|                     .collect::<Vec<String>>() | ||||
|                     .join("\n"); | ||||
|                     .join(""); | ||||
|  | ||||
|                 let mut embed = CreateEmbed::default(); | ||||
|                 embed | ||||
|   | ||||
| @@ -17,17 +17,13 @@ use regex::Regex; | ||||
|  | ||||
| lazy_static! { | ||||
|     pub static ref DEFAULT_AVATAR: AttachmentType<'static> = ( | ||||
|         include_bytes!(concat!( | ||||
|             env!("CARGO_MANIFEST_DIR"), | ||||
|             "/assets/", | ||||
|             env!("WEBHOOK_AVATAR", "WEBHOOK_AVATAR not provided for compilation") | ||||
|         )) as &[u8], | ||||
|         env!("WEBHOOK_AVATAR"), | ||||
|         include_bytes!(concat!(env!("CARGO_MANIFEST_DIR"), "/assets/webhook.jpg")) as &[u8], | ||||
|         "webhook.jpg", | ||||
|     ) | ||||
|         .into(); | ||||
|     pub static ref REGEX_CHANNEL_USER: Regex = Regex::new(r#"\s*<(#|@)(?:!)?(\d+)>\s*"#).unwrap(); | ||||
|     pub static ref SUBSCRIPTION_ROLES: HashSet<u64> = HashSet::from_iter( | ||||
|         env::var("SUBSCRIPTION_ROLES") | ||||
|         env::var("PATREON_ROLE_ID") | ||||
|             .map(|var| var | ||||
|                 .split(',') | ||||
|                 .filter_map(|item| { item.parse::<u64>().ok() }) | ||||
| @@ -35,7 +31,7 @@ lazy_static! { | ||||
|             .unwrap_or_else(|_| Vec::new()) | ||||
|     ); | ||||
|     pub static ref CNC_GUILD: Option<u64> = | ||||
|         env::var("CNC_GUILD").map(|var| var.parse::<u64>().ok()).ok().flatten(); | ||||
|         env::var("PATREON_GUILD_ID").map(|var| var.parse::<u64>().ok()).ok().flatten(); | ||||
|     pub static ref MIN_INTERVAL: i64 = | ||||
|         env::var("MIN_INTERVAL").ok().and_then(|inner| inner.parse::<i64>().ok()).unwrap_or(600); | ||||
|     pub static ref MAX_TIME: i64 = env::var("MAX_TIME") | ||||
| @@ -48,5 +44,5 @@ lazy_static! { | ||||
|         .map_or(THEME_COLOR_FALLBACK, |inner| u32::from_str_radix(&inner, 16) | ||||
|             .unwrap_or(THEME_COLOR_FALLBACK)); | ||||
|     pub static ref PYTHON_LOCATION: String = | ||||
|         env::var("PYTHON_LOCATION").unwrap_or_else(|_| "venv/bin/python3".to_string()); | ||||
|         env::var("PYTHON_LOCATION").unwrap_or_else(|_| "/usr/bin/python3".to_string()); | ||||
| } | ||||
|   | ||||
							
								
								
									
										15
									
								
								src/hooks.rs
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								src/hooks.rs
									
									
									
									
									
								
							| @@ -53,19 +53,22 @@ async fn check_self_permissions(ctx: Context<'_>) -> bool { | ||||
|             .member_permissions(&ctx.discord(), user_id) | ||||
|             .await | ||||
|             .map_or(false, |p| p.manage_webhooks()); | ||||
|  | ||||
|         let (view_channel, send_messages, embed_links) = ctx | ||||
|             .channel_id() | ||||
|             .to_channel_cached(&ctx.discord()) | ||||
|             .to_channel(&ctx.discord()) | ||||
|             .await | ||||
|             .ok() | ||||
|             .and_then(|c| { | ||||
|                 if let Channel::Guild(channel) = c { | ||||
|                     channel.permissions_for_user(&ctx.discord(), user_id).ok() | ||||
|                     let perms = channel.permissions_for_user(&ctx.discord(), user_id).ok()?; | ||||
|  | ||||
|                     Some((perms.view_channel(), perms.send_messages(), perms.embed_links())) | ||||
|                 } else { | ||||
|                     None | ||||
|                 } | ||||
|             }) | ||||
|             .map_or((false, false, false), |p| { | ||||
|                 (p.view_channel(), p.send_messages(), p.embed_links()) | ||||
|             }); | ||||
|             .unwrap_or((false, false, false)); | ||||
|  | ||||
|         if manage_webhooks && send_messages && embed_links { | ||||
|             true | ||||
| @@ -81,8 +84,8 @@ async fn check_self_permissions(ctx: Context<'_>) -> bool { | ||||
| {}     **Manage Webhooks**", | ||||
|                         if view_channel { "✅" } else { "❌" }, | ||||
|                         if send_messages { "✅" } else { "❌" }, | ||||
|                         if manage_webhooks { "✅" } else { "❌" }, | ||||
|                         if embed_links { "✅" } else { "❌" }, | ||||
|                         if manage_webhooks { "✅" } else { "❌" }, | ||||
|                     )) | ||||
|                 }) | ||||
|                 .await; | ||||
|   | ||||
| @@ -110,13 +110,14 @@ impl OverflowOp for u64 { | ||||
| #[derive(Copy, Clone)] | ||||
| pub struct Interval { | ||||
|     pub month: u64, | ||||
|     pub day: u64, | ||||
|     pub sec: u64, | ||||
| } | ||||
|  | ||||
| struct Parser<'a> { | ||||
|     iter: Chars<'a>, | ||||
|     src: &'a str, | ||||
|     current: (u64, u64, u64), | ||||
|     current: (u64, u64, u64, u64), | ||||
| } | ||||
|  | ||||
| impl<'a> Parser<'a> { | ||||
| @@ -140,17 +141,17 @@ impl<'a> Parser<'a> { | ||||
|         Ok(None) | ||||
|     } | ||||
|     fn parse_unit(&mut self, n: u64, start: usize, end: usize) -> Result<(), Error> { | ||||
|         let (mut month, mut sec, nsec) = match &self.src[start..end] { | ||||
|             "nanos" | "nsec" | "ns" => (0u64, 0u64, n), | ||||
|             "usec" | "us" => (0, 0u64, n.mul(1000)?), | ||||
|             "millis" | "msec" | "ms" => (0, 0u64, n.mul(1_000_000)?), | ||||
|             "seconds" | "second" | "secs" | "sec" | "s" => (0, n, 0), | ||||
|             "minutes" | "minute" | "min" | "mins" | "m" => (0, n.mul(60)?, 0), | ||||
|             "hours" | "hour" | "hr" | "hrs" | "h" => (0, n.mul(3600)?, 0), | ||||
|             "days" | "day" | "d" => (0, n.mul(86400)?, 0), | ||||
|             "weeks" | "week" | "w" => (0, n.mul(86400 * 7)?, 0), | ||||
|             "months" | "month" | "M" => (n, 0, 0), | ||||
|             "years" | "year" | "y" => (12, 0, 0), | ||||
|         let (mut month, mut day, mut sec, nsec) = match &self.src[start..end] { | ||||
|             "nanos" | "nsec" | "ns" => (0, 0u64, 0u64, n), | ||||
|             "usec" | "us" => (0, 0, 0u64, n.mul(1000)?), | ||||
|             "millis" | "msec" | "ms" => (0, 0, 0u64, n.mul(1_000_000)?), | ||||
|             "seconds" | "second" | "secs" | "sec" | "s" => (0, 0, n, 0), | ||||
|             "minutes" | "minute" | "min" | "mins" | "m" => (0, 0, n.mul(60)?, 0), | ||||
|             "hours" | "hour" | "hr" | "hrs" | "h" => (0, 0, n.mul(3600)?, 0), | ||||
|             "days" | "day" | "d" => (0, n, 0, 0), | ||||
|             "weeks" | "week" | "w" => (0, n.mul(7)?, 0, 0), | ||||
|             "months" | "month" | "M" => (n, 0, 0, 0), | ||||
|             "years" | "year" | "y" => (n.mul(12)?, 0, 0, 0), | ||||
|             _ => { | ||||
|                 return Err(Error::UnknownUnit { | ||||
|                     start, | ||||
| @@ -160,15 +161,16 @@ impl<'a> Parser<'a> { | ||||
|                 }); | ||||
|             } | ||||
|         }; | ||||
|         let mut nsec = self.current.2 + nsec; | ||||
|         let mut nsec = self.current.3 + nsec; | ||||
|         if nsec > 1_000_000_000 { | ||||
|             sec += nsec / 1_000_000_000; | ||||
|             nsec %= 1_000_000_000; | ||||
|         } | ||||
|         sec += self.current.1; | ||||
|         sec += self.current.2; | ||||
|         day += self.current.1; | ||||
|         month += self.current.0; | ||||
|  | ||||
|         self.current = (month, sec, nsec); | ||||
|         self.current = (month, day, sec, nsec); | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
| @@ -215,7 +217,13 @@ impl<'a> Parser<'a> { | ||||
|             self.parse_unit(n, start, off)?; | ||||
|             n = match self.parse_first_char()? { | ||||
|                 Some(n) => n, | ||||
|                 None => return Ok(Interval { month: self.current.0, sec: self.current.1 }), | ||||
|                 None => { | ||||
|                     return Ok(Interval { | ||||
|                         month: self.current.0, | ||||
|                         day: self.current.1, | ||||
|                         sec: self.current.2, | ||||
|                     }) | ||||
|                 } | ||||
|             }; | ||||
|         } | ||||
|     } | ||||
| @@ -247,5 +255,73 @@ impl<'a> Parser<'a> { | ||||
| /// assert_eq!(parse_duration("32ms"), Ok(Duration::new(0, 32_000_000))); | ||||
| /// ``` | ||||
| pub fn parse_duration(s: &str) -> Result<Interval, Error> { | ||||
|     Parser { iter: s.chars(), src: s, current: (0, 0, 0) }.parse() | ||||
|     Parser { iter: s.chars(), src: s, current: (0, 0, 0, 0) }.parse() | ||||
| } | ||||
|  | ||||
| #[cfg(test)] | ||||
| mod tests { | ||||
|     use super::*; | ||||
|  | ||||
|     #[test] | ||||
|     fn parse_seconds() { | ||||
|         let interval = parse_duration("10 seconds").unwrap(); | ||||
|  | ||||
|         assert_eq!(interval.sec, 10); | ||||
|         assert_eq!(interval.day, 0); | ||||
|         assert_eq!(interval.month, 0); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn parse_minutes() { | ||||
|         let interval = parse_duration("10 minutes").unwrap(); | ||||
|  | ||||
|         assert_eq!(interval.sec, 600); | ||||
|         assert_eq!(interval.day, 0); | ||||
|         assert_eq!(interval.month, 0); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn parse_hours() { | ||||
|         let interval = parse_duration("10 hours").unwrap(); | ||||
|  | ||||
|         assert_eq!(interval.sec, 36_000); | ||||
|         assert_eq!(interval.day, 0); | ||||
|         assert_eq!(interval.month, 0); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn parse_days() { | ||||
|         let interval = parse_duration("10 days").unwrap(); | ||||
|  | ||||
|         assert_eq!(interval.sec, 0); | ||||
|         assert_eq!(interval.day, 10); | ||||
|         assert_eq!(interval.month, 0); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn parse_weeks() { | ||||
|         let interval = parse_duration("10 weeks").unwrap(); | ||||
|  | ||||
|         assert_eq!(interval.sec, 0); | ||||
|         assert_eq!(interval.day, 70); | ||||
|         assert_eq!(interval.month, 0); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn parse_months() { | ||||
|         let interval = parse_duration("10 months").unwrap(); | ||||
|  | ||||
|         assert_eq!(interval.sec, 0); | ||||
|         assert_eq!(interval.day, 0); | ||||
|         assert_eq!(interval.month, 10); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn parse_years() { | ||||
|         let interval = parse_duration("10 years").unwrap(); | ||||
|  | ||||
|         assert_eq!(interval.sec, 0); | ||||
|         assert_eq!(interval.day, 0); | ||||
|         assert_eq!(interval.month, 120); | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										13
									
								
								src/main.rs
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								src/main.rs
									
									
									
									
									
								
							| @@ -18,10 +18,10 @@ use std::{ | ||||
|     env, | ||||
|     error::Error as StdError, | ||||
|     fmt::{Debug, Display, Formatter}, | ||||
|     path::Path, | ||||
| }; | ||||
|  | ||||
| use chrono_tz::Tz; | ||||
| use dotenv::dotenv; | ||||
| use log::{error, warn}; | ||||
| use poise::serenity_prelude::model::{ | ||||
|     gateway::GatewayIntents, | ||||
| @@ -75,7 +75,7 @@ impl Display for Ended { | ||||
|  | ||||
| impl StdError for Ended {} | ||||
|  | ||||
| #[tokio::main] | ||||
| #[tokio::main(flavor = "multi_thread")] | ||||
| async fn main() -> Result<(), Box<dyn StdError + Send + Sync>> { | ||||
|     let (tx, mut rx) = broadcast::channel(16); | ||||
|  | ||||
| @@ -88,7 +88,11 @@ async fn main() -> Result<(), Box<dyn StdError + Send + Sync>> { | ||||
| async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | ||||
|     env_logger::init(); | ||||
|  | ||||
|     dotenv()?; | ||||
|     if Path::new("/etc/reminder-rs/config.env").exists() { | ||||
|         dotenv::from_path("/etc/reminder-rs/config.env")?; | ||||
|     } else { | ||||
|         dotenv::from_path(".env")?; | ||||
|     } | ||||
|  | ||||
|     let discord_token = env::var("DISCORD_TOKEN").expect("Missing DISCORD_TOKEN from environment"); | ||||
|  | ||||
| @@ -133,6 +137,7 @@ async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | ||||
|                 ], | ||||
|                 ..reminder_cmds::timer_base() | ||||
|             }, | ||||
|             reminder_cmds::multiline(), | ||||
|             reminder_cmds::remind(), | ||||
|             poise::Command { | ||||
|                 subcommands: vec![ | ||||
| @@ -167,6 +172,8 @@ async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | ||||
|     let database = | ||||
|         Pool::connect(&env::var("DATABASE_URL").expect("No database URL provided")).await.unwrap(); | ||||
|  | ||||
|     sqlx::migrate!().run(&database).await?; | ||||
|  | ||||
|     let popular_timezones = sqlx::query!( | ||||
|         "SELECT IFNULL(timezone, 'UTC') AS timezone | ||||
|         FROM users | ||||
|   | ||||
| @@ -5,7 +5,7 @@ pub mod timer; | ||||
| pub mod user_data; | ||||
|  | ||||
| use chrono_tz::Tz; | ||||
| use poise::serenity_prelude::{async_trait, model::id::UserId}; | ||||
| use poise::serenity_prelude::{async_trait, model::id::UserId, ChannelType}; | ||||
|  | ||||
| use crate::{ | ||||
|     models::{channel_data::ChannelData, user_data::UserData}, | ||||
| @@ -43,7 +43,20 @@ impl CtxData for Context<'_> { | ||||
|     } | ||||
|  | ||||
|     async fn channel_data(&self) -> Result<ChannelData, Box<dyn std::error::Error + Sync + Send>> { | ||||
|         let channel = self.channel_id().to_channel_cached(&self.discord()).unwrap(); | ||||
|         // If we're in a thread, get the parent channel. | ||||
|         let recv_channel = self.channel_id().to_channel(&self.discord()).await?; | ||||
|  | ||||
|         let channel = match recv_channel.guild() { | ||||
|             Some(guild_channel) => { | ||||
|                 if guild_channel.kind == ChannelType::PublicThread { | ||||
|                     guild_channel.parent_id.unwrap().to_channel_cached(&self.discord()).unwrap() | ||||
|                 } else { | ||||
|                     self.channel_id().to_channel_cached(&self.discord()).unwrap() | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             None => self.channel_id().to_channel_cached(&self.discord()).unwrap(), | ||||
|         }; | ||||
|  | ||||
|         ChannelData::from_channel(&channel, &self.data().database).await | ||||
|     } | ||||
|   | ||||
| @@ -9,7 +9,7 @@ use poise::serenity_prelude::{ | ||||
|         id::{ChannelId, GuildId, UserId}, | ||||
|         webhook::Webhook, | ||||
|     }, | ||||
|     Result as SerenityResult, | ||||
|     ChannelType, Result as SerenityResult, | ||||
| }; | ||||
| use sqlx::MySqlPool; | ||||
|  | ||||
| @@ -51,9 +51,11 @@ pub struct ReminderBuilder { | ||||
|     pool: MySqlPool, | ||||
|     uid: String, | ||||
|     channel: u32, | ||||
|     thread_id: Option<u64>, | ||||
|     utc_time: NaiveDateTime, | ||||
|     timezone: String, | ||||
|     interval_secs: Option<i64>, | ||||
|     interval_seconds: Option<i64>, | ||||
|     interval_days: Option<i64>, | ||||
|     interval_months: Option<i64>, | ||||
|     expires: Option<NaiveDateTime>, | ||||
|     content: String, | ||||
| @@ -87,6 +89,7 @@ INSERT INTO reminders ( | ||||
|     `utc_time`, | ||||
|     `timezone`, | ||||
|     `interval_seconds`, | ||||
|     `interval_days`, | ||||
|     `interval_months`, | ||||
|     `expires`, | ||||
|     `content`, | ||||
| @@ -106,6 +109,7 @@ INSERT INTO reminders ( | ||||
|     ?, | ||||
|     ?, | ||||
|     ?, | ||||
|     ?, | ||||
|     ? | ||||
| ) | ||||
|             ", | ||||
| @@ -113,7 +117,8 @@ INSERT INTO reminders ( | ||||
|                         self.channel, | ||||
|                         utc_time, | ||||
|                         self.timezone, | ||||
|                         self.interval_secs, | ||||
|                         self.interval_seconds, | ||||
|                         self.interval_days, | ||||
|                         self.interval_months, | ||||
|                         self.expires, | ||||
|                         self.content, | ||||
| @@ -175,17 +180,15 @@ impl<'a> MultiReminderBuilder<'a> { | ||||
|     } | ||||
|  | ||||
|     pub fn time<T: Into<i64>>(mut self, time: T) -> Self { | ||||
|         self.utc_time = NaiveDateTime::from_timestamp(time.into(), 0); | ||||
|         if let Some(utc_time) = NaiveDateTime::from_timestamp_opt(time.into(), 0) { | ||||
|             self.utc_time = utc_time; | ||||
|         } | ||||
|  | ||||
|         self | ||||
|     } | ||||
|  | ||||
|     pub fn expires<T: Into<i64>>(mut self, time: Option<T>) -> Self { | ||||
|         if let Some(t) = time { | ||||
|             self.expires = Some(NaiveDateTime::from_timestamp(t.into(), 0)); | ||||
|         } else { | ||||
|             self.expires = None; | ||||
|         } | ||||
|         self.expires = time.map(|t| NaiveDateTime::from_timestamp_opt(t.into(), 0)).flatten(); | ||||
|  | ||||
|         self | ||||
|     } | ||||
| @@ -212,13 +215,19 @@ impl<'a> MultiReminderBuilder<'a> { | ||||
|  | ||||
|         let mut ok_locs = HashSet::new(); | ||||
|  | ||||
|         if self.interval.map_or(false, |i| ((i.sec + i.month * 30 * DAY) as i64) < *MIN_INTERVAL) { | ||||
|         if self | ||||
|             .interval | ||||
|             .map_or(false, |i| ((i.sec + i.day * DAY + i.month * 30 * DAY) as i64) < *MIN_INTERVAL) | ||||
|         { | ||||
|             errors.insert(ReminderError::ShortInterval); | ||||
|         } else if self.interval.map_or(false, |i| ((i.sec + i.month * 30 * DAY) as i64) > *MAX_TIME) | ||||
|         } else if self | ||||
|             .interval | ||||
|             .map_or(false, |i| ((i.sec + i.day * DAY + i.month * 30 * DAY) as i64) > *MAX_TIME) | ||||
|         { | ||||
|             errors.insert(ReminderError::LongInterval); | ||||
|         } else { | ||||
|             for scope in self.scopes { | ||||
|                 let thread_id = None; | ||||
|                 let db_channel_id = match scope { | ||||
|                     ReminderScope::User(user_id) => { | ||||
|                         if let Ok(user) = UserId(user_id).to_user(&self.ctx.discord()).await { | ||||
| @@ -251,14 +260,29 @@ impl<'a> MultiReminderBuilder<'a> { | ||||
|                         let channel = | ||||
|                             ChannelId(channel_id).to_channel(&self.ctx.discord()).await.unwrap(); | ||||
|  | ||||
|                         if let Some(guild_channel) = channel.clone().guild() { | ||||
|                         if let Some(mut guild_channel) = channel.clone().guild() { | ||||
|                             if Some(guild_channel.guild_id) != self.guild_id { | ||||
|                                 Err(ReminderError::InvalidTag) | ||||
|                             } else { | ||||
|                                 let mut channel_data = | ||||
|                                     ChannelData::from_channel(&channel, &self.ctx.data().database) | ||||
|                                 let mut channel_data = if guild_channel.kind | ||||
|                                     == ChannelType::PublicThread | ||||
|                                 { | ||||
|                                     // fixme jesus christ | ||||
|                                     let parent = guild_channel | ||||
|                                         .parent_id | ||||
|                                         .unwrap() | ||||
|                                         .to_channel(&self.ctx.discord()) | ||||
|                                         .await | ||||
|                                         .unwrap(); | ||||
|                                     guild_channel = parent.clone().guild().unwrap(); | ||||
|                                     ChannelData::from_channel(&parent, &self.ctx.data().database) | ||||
|                                         .await | ||||
|                                         .unwrap() | ||||
|                                 } else { | ||||
|                                     ChannelData::from_channel(&channel, &self.ctx.data().database) | ||||
|                                         .await | ||||
|                                         .unwrap() | ||||
|                                 }; | ||||
|  | ||||
|                                 if channel_data.webhook_id.is_none() | ||||
|                                     || channel_data.webhook_token.is_none() | ||||
| @@ -300,9 +324,11 @@ impl<'a> MultiReminderBuilder<'a> { | ||||
|                             pool: self.ctx.data().database.clone(), | ||||
|                             uid: generate_uid(), | ||||
|                             channel: c, | ||||
|                             thread_id, | ||||
|                             utc_time: self.utc_time, | ||||
|                             timezone: self.timezone.to_string(), | ||||
|                             interval_secs: self.interval.map(|i| i.sec as i64), | ||||
|                             interval_seconds: self.interval.map(|i| i.sec as i64), | ||||
|                             interval_days: self.interval.map(|i| i.day as i64), | ||||
|                             interval_months: self.interval.map(|i| i.month as i64), | ||||
|                             expires: self.expires, | ||||
|                             content: self.content.content.clone(), | ||||
|   | ||||
| @@ -6,7 +6,7 @@ pub mod look_flags; | ||||
|  | ||||
| use std::hash::{Hash, Hasher}; | ||||
|  | ||||
| use chrono::{NaiveDateTime, TimeZone}; | ||||
| use chrono::{DateTime, NaiveDateTime, Utc}; | ||||
| use chrono_tz::Tz; | ||||
| use poise::serenity_prelude::{ | ||||
|     model::id::{ChannelId, GuildId, UserId}, | ||||
| @@ -24,8 +24,9 @@ pub struct Reminder { | ||||
|     pub id: u32, | ||||
|     pub uid: String, | ||||
|     pub channel: u64, | ||||
|     pub utc_time: NaiveDateTime, | ||||
|     pub utc_time: DateTime<Utc>, | ||||
|     pub interval_seconds: Option<u32>, | ||||
|     pub interval_days: Option<u32>, | ||||
|     pub interval_months: Option<u32>, | ||||
|     pub expires: Option<NaiveDateTime>, | ||||
|     pub enabled: bool, | ||||
| @@ -59,6 +60,7 @@ SELECT | ||||
|     channels.channel, | ||||
|     reminders.utc_time, | ||||
|     reminders.interval_seconds, | ||||
|     reminders.interval_days, | ||||
|     reminders.interval_months, | ||||
|     reminders.expires, | ||||
|     reminders.enabled, | ||||
| @@ -95,6 +97,7 @@ SELECT | ||||
|     channels.channel, | ||||
|     reminders.utc_time, | ||||
|     reminders.interval_seconds, | ||||
|     reminders.interval_days, | ||||
|     reminders.interval_months, | ||||
|     reminders.expires, | ||||
|     reminders.enabled, | ||||
| @@ -138,6 +141,7 @@ SELECT | ||||
|     channels.channel, | ||||
|     reminders.utc_time, | ||||
|     reminders.interval_seconds, | ||||
|     reminders.interval_days, | ||||
|     reminders.interval_months, | ||||
|     reminders.expires, | ||||
|     reminders.enabled, | ||||
| @@ -195,6 +199,7 @@ SELECT | ||||
|     channels.channel, | ||||
|     reminders.utc_time, | ||||
|     reminders.interval_seconds, | ||||
|     reminders.interval_days, | ||||
|     reminders.interval_months, | ||||
|     reminders.expires, | ||||
|     reminders.enabled, | ||||
| @@ -228,6 +233,7 @@ SELECT | ||||
|     channels.channel, | ||||
|     reminders.utc_time, | ||||
|     reminders.interval_seconds, | ||||
|     reminders.interval_days, | ||||
|     reminders.interval_months, | ||||
|     reminders.expires, | ||||
|     reminders.enabled, | ||||
| @@ -262,6 +268,7 @@ SELECT | ||||
|     channels.channel, | ||||
|     reminders.utc_time, | ||||
|     reminders.interval_seconds, | ||||
|     reminders.interval_days, | ||||
|     reminders.interval_months, | ||||
|     reminders.expires, | ||||
|     reminders.enabled, | ||||
| @@ -310,30 +317,32 @@ WHERE | ||||
|             count + 1, | ||||
|             self.display_content(), | ||||
|             self.channel, | ||||
|             timezone.timestamp(self.utc_time.timestamp(), 0).format("%Y-%m-%d %H:%M:%S") | ||||
|             self.utc_time.with_timezone(timezone).format("%Y-%m-%d %H:%M:%S") | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     pub fn display(&self, flags: &LookFlags, timezone: &Tz) -> String { | ||||
|         let time_display = match flags.time_display { | ||||
|             TimeDisplayType::Absolute => timezone | ||||
|                 .timestamp(self.utc_time.timestamp(), 0) | ||||
|                 .format("%Y-%m-%d %H:%M:%S") | ||||
|                 .to_string(), | ||||
|             TimeDisplayType::Absolute => { | ||||
|                 self.utc_time.with_timezone(timezone).format("%Y-%m-%d %H:%M:%S").to_string() | ||||
|             } | ||||
|  | ||||
|             TimeDisplayType::Relative => format!("<t:{}:R>", self.utc_time.timestamp()), | ||||
|         }; | ||||
|  | ||||
|         if self.interval_seconds.is_some() || self.interval_months.is_some() { | ||||
|         if self.interval_seconds.is_some() | ||||
|             || self.interval_days.is_some() | ||||
|             || self.interval_months.is_some() | ||||
|         { | ||||
|             format!( | ||||
|                 "'{}' *occurs next at* **{}**, repeating (set by {})", | ||||
|                 "'{}' *occurs next at* **{}**, repeating (set by {})\n", | ||||
|                 self.display_content(), | ||||
|                 time_display, | ||||
|                 self.set_by.map(|i| format!("<@{}>", i)).unwrap_or_else(|| "unknown".to_string()) | ||||
|             ) | ||||
|         } else { | ||||
|             format!( | ||||
|                 "'{}' *occurs next at* **{}** (set by {})", | ||||
|                 "'{}' *occurs next at* **{}** (set by {})\n", | ||||
|                 self.display_content(), | ||||
|                 time_display, | ||||
|                 self.set_by.map(|i| format!("<@{}>", i)).unwrap_or_else(|| "unknown".to_string()) | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| use chrono::NaiveDateTime; | ||||
| use chrono::{DateTime, Utc}; | ||||
| use sqlx::MySqlPool; | ||||
|  | ||||
| pub struct Timer { | ||||
|     pub name: String, | ||||
|     pub start_time: NaiveDateTime, | ||||
|     pub start_time: DateTime<Utc>, | ||||
|     pub owner: u64, | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										13
									
								
								systemd/reminder-rs.service
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								systemd/reminder-rs.service
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| [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(); | ||||
|     pub static ref SUBSCRIPTION_ROLES: HashSet<u64> = HashSet::from_iter( | ||||
|         env::var("SUBSCRIPTION_ROLES") | ||||
|         env::var("PATREON_ROLE_ID") | ||||
|             .map(|var| var | ||||
|                 .split(',') | ||||
|                 .filter_map(|item| { item.parse::<u64>().ok() }) | ||||
| @@ -39,7 +39,7 @@ lazy_static! { | ||||
|             .unwrap_or_else(|_| Vec::new()) | ||||
|     ); | ||||
|     pub static ref CNC_GUILD: Option<u64> = | ||||
|         env::var("CNC_GUILD").map(|var| var.parse::<u64>().ok()).ok().flatten(); | ||||
|         env::var("PATREON_GUILD_ID").map(|var| var.parse::<u64>().ok()).ok().flatten(); | ||||
|     pub static ref MIN_INTERVAL: u32 = env::var("MIN_INTERVAL") | ||||
|         .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_SECRET").expect("`OAUTH2_CLIENT_SECRET' not supplied"); | ||||
|     env::var("OAUTH2_DISCORD_CALLBACK").expect("`OAUTH2_DISCORD_CALLBACK' not supplied"); | ||||
|     env::var("PATREON_GUILD_ID").expect("`PATREON_GUILD' not supplied"); | ||||
|     env::var("PATREON_GUILD_ID").expect("`PATREON_GUILD_ID' not supplied"); | ||||
|     info!("Done!"); | ||||
|  | ||||
|     let oauth2_client = BasicClient::new( | ||||
|   | ||||
| @@ -58,6 +58,7 @@ pub async fn export_reminders( | ||||
|                  reminders.enabled, | ||||
|                  reminders.expires, | ||||
|                  reminders.interval_seconds, | ||||
|                  reminders.interval_days, | ||||
|                  reminders.interval_months, | ||||
|                  reminders.name, | ||||
|                  reminders.restartable, | ||||
| @@ -159,6 +160,7 @@ pub async fn import_reminders( | ||||
|                                     enabled: record.enabled, | ||||
|                                     expires: record.expires, | ||||
|                                     interval_seconds: record.interval_seconds, | ||||
|                                     interval_days: record.interval_days, | ||||
|                                     interval_months: record.interval_months, | ||||
|                                     name: record.name, | ||||
|                                     restartable: record.restartable, | ||||
| @@ -318,13 +320,6 @@ 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!( | ||||
|                     "INSERT INTO todos (value, channel_id, guild_id) VALUES {}", | ||||
|                     vec![query_placeholder].repeat(query_params.len()).join(",") | ||||
|   | ||||
| @@ -16,10 +16,12 @@ use serenity::{ | ||||
| use sqlx::{MySql, Pool}; | ||||
|  | ||||
| use crate::{ | ||||
|     check_guild_subscription, check_subscription, | ||||
|     consts::{ | ||||
|         MAX_CONTENT_LENGTH, MAX_EMBED_AUTHOR_LENGTH, MAX_EMBED_DESCRIPTION_LENGTH, | ||||
|         MAX_EMBED_FIELDS, MAX_EMBED_FIELD_TITLE_LENGTH, MAX_EMBED_FIELD_VALUE_LENGTH, | ||||
|         MAX_EMBED_FOOTER_LENGTH, MAX_EMBED_TITLE_LENGTH, MAX_URL_LENGTH, MAX_USERNAME_LENGTH, | ||||
|         MIN_INTERVAL, | ||||
|     }, | ||||
|     routes::dashboard::{ | ||||
|         create_database_channel, create_reminder, template_name_default, DeleteReminder, | ||||
| @@ -247,9 +249,9 @@ pub async fn create_reminder_template( | ||||
|             Ok(json!({})) | ||||
|         } | ||||
|         Err(e) => { | ||||
|             warn!("Could not fetch templates from {}: {:?}", id, e); | ||||
|             warn!("Could not create template for {}: {:?}", id, e); | ||||
|  | ||||
|             json_err!("Could not get templates") | ||||
|             json_err!("Could not create template") | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -339,6 +341,7 @@ pub async fn get_reminders(id: u64, ctx: &State<Context>, pool: &State<Pool<MySq | ||||
|                  reminders.enabled, | ||||
|                  reminders.expires, | ||||
|                  reminders.interval_seconds, | ||||
|                  reminders.interval_days, | ||||
|                  reminders.interval_months, | ||||
|                  reminders.name, | ||||
|                  reminders.restartable, | ||||
| @@ -374,35 +377,109 @@ pub async fn edit_reminder( | ||||
|     reminder: Json<PatchReminder>, | ||||
|     serenity_context: &State<Context>, | ||||
|     pool: &State<Pool<MySql>>, | ||||
|     cookies: &CookieJar<'_>, | ||||
| ) -> JsonResult { | ||||
|     check_authorization!(cookies, serenity_context.inner(), id); | ||||
|  | ||||
|     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.[ | ||||
|         attachment, | ||||
|         attachment_name, | ||||
|         avatar, | ||||
|         content, | ||||
|         embed_author, | ||||
|         embed_author_url, | ||||
|         embed_color, | ||||
|         embed_description, | ||||
|         embed_footer, | ||||
|         embed_footer_url, | ||||
|         embed_image_url, | ||||
|         embed_thumbnail_url, | ||||
|         embed_title, | ||||
|         embed_fields, | ||||
|         enabled, | ||||
|         expires, | ||||
|         interval_seconds, | ||||
|         interval_months, | ||||
|         name, | ||||
|         restartable, | ||||
|         tts, | ||||
|         username, | ||||
|         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 { | ||||
|         let channel = ChannelId(reminder.channel).to_channel_cached(&serenity_context.inner()); | ||||
|         match channel { | ||||
| @@ -483,6 +560,7 @@ pub async fn edit_reminder( | ||||
|          reminders.enabled, | ||||
|          reminders.expires, | ||||
|          reminders.interval_seconds, | ||||
|          reminders.interval_days, | ||||
|          reminders.interval_months, | ||||
|          reminders.name, | ||||
|          reminders.restartable, | ||||
|   | ||||
| @@ -8,13 +8,13 @@ use rocket::{ | ||||
|     serde::json::{json, Value as JsonValue}, | ||||
| }; | ||||
| use rocket_dyn_templates::Template; | ||||
| use serde::{Deserialize, Serialize}; | ||||
| use serde::{Deserialize, Deserializer, Serialize}; | ||||
| use serenity::{ | ||||
|     client::Context, | ||||
|     http::Http, | ||||
|     model::id::{ChannelId, GuildId, UserId}, | ||||
| }; | ||||
| use sqlx::{types::Json, Executor, MySql, Pool}; | ||||
| use sqlx::{types::Json, Executor}; | ||||
|  | ||||
| use crate::{ | ||||
|     check_guild_subscription, check_subscription, | ||||
| @@ -50,6 +50,18 @@ fn id_default() -> u32 { | ||||
|     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)] | ||||
| pub struct ReminderTemplate { | ||||
|     #[serde(default = "id_default")] | ||||
| @@ -132,6 +144,7 @@ pub struct Reminder { | ||||
|     enabled: bool, | ||||
|     expires: Option<NaiveDateTime>, | ||||
|     interval_seconds: Option<u32>, | ||||
|     interval_days: Option<u32>, | ||||
|     interval_months: Option<u32>, | ||||
|     #[serde(default = "name_default")] | ||||
|     name: String, | ||||
| @@ -164,6 +177,7 @@ pub struct ReminderCsv { | ||||
|     enabled: bool, | ||||
|     expires: Option<NaiveDateTime>, | ||||
|     interval_seconds: Option<u32>, | ||||
|     interval_days: Option<u32>, | ||||
|     interval_months: Option<u32>, | ||||
|     #[serde(default = "name_default")] | ||||
|     name: String, | ||||
| @@ -177,10 +191,13 @@ pub struct ReminderCsv { | ||||
| pub struct PatchReminder { | ||||
|     uid: String, | ||||
|     #[serde(default)] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     attachment: Unset<Option<String>>, | ||||
|     #[serde(default)] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     attachment_name: Unset<Option<String>>, | ||||
|     #[serde(default)] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     avatar: Unset<Option<String>>, | ||||
|     #[serde(default = "channel_default")] | ||||
|     #[serde(with = "string")] | ||||
| @@ -190,6 +207,7 @@ pub struct PatchReminder { | ||||
|     #[serde(default)] | ||||
|     embed_author: Unset<String>, | ||||
|     #[serde(default)] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     embed_author_url: Unset<Option<String>>, | ||||
|     #[serde(default)] | ||||
|     embed_color: Unset<u32>, | ||||
| @@ -198,10 +216,13 @@ pub struct PatchReminder { | ||||
|     #[serde(default)] | ||||
|     embed_footer: Unset<String>, | ||||
|     #[serde(default)] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     embed_footer_url: Unset<Option<String>>, | ||||
|     #[serde(default)] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     embed_image_url: Unset<Option<String>>, | ||||
|     #[serde(default)] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     embed_thumbnail_url: Unset<Option<String>>, | ||||
|     #[serde(default)] | ||||
|     embed_title: Unset<String>, | ||||
| @@ -210,10 +231,16 @@ pub struct PatchReminder { | ||||
|     #[serde(default)] | ||||
|     enabled: Unset<bool>, | ||||
|     #[serde(default)] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     expires: Unset<Option<NaiveDateTime>>, | ||||
|     #[serde(default)] | ||||
|     #[serde(default = "interval_default")] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     interval_seconds: Unset<Option<u32>>, | ||||
|     #[serde(default)] | ||||
|     #[serde(default = "interval_default")] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     interval_days: Unset<Option<u32>>, | ||||
|     #[serde(default = "interval_default")] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     interval_months: Unset<Option<u32>>, | ||||
|     #[serde(default)] | ||||
|     name: Unset<String>, | ||||
| @@ -222,11 +249,36 @@ pub struct PatchReminder { | ||||
|     #[serde(default)] | ||||
|     tts: Unset<bool>, | ||||
|     #[serde(default)] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     username: Unset<Option<String>>, | ||||
|     #[serde(default)] | ||||
|     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 { | ||||
|     let mut generator: OsRng = Default::default(); | ||||
|  | ||||
| @@ -301,11 +353,28 @@ pub struct TodoCsv { | ||||
|  | ||||
| pub async fn create_reminder( | ||||
|     ctx: &Context, | ||||
|     pool: &Pool<MySql>, | ||||
|     pool: impl sqlx::Executor<'_, Database = Database> + Copy, | ||||
|     guild_id: GuildId, | ||||
|     user_id: UserId, | ||||
|     reminder: Reminder, | ||||
| ) -> JsonResult { | ||||
|     // check guild in db | ||||
|     match sqlx::query!("SELECT 1 as A FROM guilds WHERE guild = ?", guild_id.0) | ||||
|         .fetch_one(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 | ||||
|     let channel = ChannelId(reminder.channel).to_channel_cached(&ctx); | ||||
|     let channel_exists = channel.is_some(); | ||||
| @@ -370,8 +439,12 @@ pub async fn create_reminder( | ||||
|     if reminder.utc_time < Utc::now().naive_utc() { | ||||
|         return Err(json!({"error": "Time must be in the future"})); | ||||
|     } | ||||
|     if reminder.interval_seconds.is_some() || reminder.interval_months.is_some() { | ||||
|     if reminder.interval_seconds.is_some() | ||||
|         || reminder.interval_days.is_some() | ||||
|         || reminder.interval_months.is_some() | ||||
|     { | ||||
|         if reminder.interval_months.unwrap_or(0) * 30 * DAY as u32 | ||||
|             + reminder.interval_days.unwrap_or(0) * DAY as u32 | ||||
|             + reminder.interval_seconds.unwrap_or(0) | ||||
|             < *MIN_INTERVAL | ||||
|         { | ||||
| @@ -380,7 +453,10 @@ pub async fn create_reminder( | ||||
|     } | ||||
|  | ||||
|     // check patreon if necessary | ||||
|     if reminder.interval_seconds.is_some() || reminder.interval_months.is_some() { | ||||
|     if reminder.interval_seconds.is_some() | ||||
|         || reminder.interval_days.is_some() | ||||
|         || reminder.interval_months.is_some() | ||||
|     { | ||||
|         if !check_guild_subscription(&ctx, guild_id).await | ||||
|             && !check_subscription(&ctx, user_id).await | ||||
|         { | ||||
| @@ -391,6 +467,11 @@ pub async fn create_reminder( | ||||
|     // base64 decode error dropped here | ||||
|     let attachment_data = reminder.attachment.as_ref().map(|s| base64::decode(s).ok()).flatten(); | ||||
|     let name = if reminder.name.is_empty() { name_default() } else { reminder.name.clone() }; | ||||
|     let username = if reminder.username.as_ref().map(|s| s.is_empty()).unwrap_or(true) { | ||||
|         None | ||||
|     } else { | ||||
|         reminder.username | ||||
|     }; | ||||
|  | ||||
|     let new_uid = generate_uid(); | ||||
|  | ||||
| @@ -416,13 +497,14 @@ pub async fn create_reminder( | ||||
|          enabled, | ||||
|          expires, | ||||
|          interval_seconds, | ||||
|          interval_days, | ||||
|          interval_months, | ||||
|          name, | ||||
|          restartable, | ||||
|          tts, | ||||
|          username, | ||||
|          `utc_time` | ||||
|         ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)", | ||||
|         ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)", | ||||
|         new_uid, | ||||
|         attachment_data, | ||||
|         reminder.attachment_name, | ||||
| @@ -442,11 +524,12 @@ pub async fn create_reminder( | ||||
|         reminder.enabled, | ||||
|         reminder.expires, | ||||
|         reminder.interval_seconds, | ||||
|         reminder.interval_days, | ||||
|         reminder.interval_months, | ||||
|         name, | ||||
|         reminder.restartable, | ||||
|         reminder.tts, | ||||
|         reminder.username, | ||||
|         username, | ||||
|         reminder.utc_time, | ||||
|     ) | ||||
|     .execute(pool) | ||||
| @@ -473,6 +556,7 @@ pub async fn create_reminder( | ||||
|              reminders.enabled, | ||||
|              reminders.expires, | ||||
|              reminders.interval_seconds, | ||||
|              reminders.interval_days, | ||||
|              reminders.interval_months, | ||||
|              reminders.name, | ||||
|              reminders.restartable, | ||||
|   | ||||
| @@ -135,14 +135,14 @@ pub async fn discord_callback( | ||||
|                     Err(Flash::new( | ||||
|                         Redirect::to(uri!(super::return_to_same_site(""))), | ||||
|                         "warning", | ||||
|                         "Your login request was rejected", | ||||
|                         "Your login request was rejected. The server may be misconfigured. Please retry or alert us in Discord.", | ||||
|                     )) | ||||
|                 } | ||||
|             } | ||||
|         } else { | ||||
|             Err(Flash::new(Redirect::to(uri!(super::return_to_same_site(""))), "danger", "Your request failed to validate, and so has been rejected (error: CSRF Validation Failure)")) | ||||
|             Err(Flash::new(Redirect::to(uri!(super::return_to_same_site(""))), "danger", "Your request failed to validate, and so has been rejected (CSRF Validation Failure)")) | ||||
|         } | ||||
|     } else { | ||||
|         Err(Flash::new(Redirect::to(uri!(super::return_to_same_site(""))), "warning", "Your request was missing information, and so has been rejected (error: CSRF Validation Tokens Missing)")) | ||||
|         Err(Flash::new(Redirect::to(uri!(super::return_to_same_site(""))), "warning", "Your request was missing information, and so has been rejected (CSRF Validation Tokens Missing)")) | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -7,8 +7,8 @@ function get_interval(element) { | ||||
|  | ||||
|     return { | ||||
|         months: parseInt(months) || null, | ||||
|         days: parseInt(days) || null, | ||||
|         seconds: | ||||
|             (parseInt(days) || 0) * 86400 + | ||||
|             (parseInt(hours) || 0) * 3600 + | ||||
|                 (parseInt(minutes) || 0) * 60 + | ||||
|                 (parseInt(seconds) || 0) || null, | ||||
| @@ -22,6 +22,15 @@ function update_interval(element) { | ||||
|     let minutes = element.querySelector('input[name="interval_minutes"]'); | ||||
|     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"); | ||||
|         days.value = days.value.padStart(1, "0"); | ||||
|         hours.value = hours.value.padStart(2, "0"); | ||||
| @@ -33,7 +42,10 @@ function update_interval(element) { | ||||
|             let remainder = seconds.value % 60; | ||||
|  | ||||
|             seconds.value = String(remainder).padStart(2, "0"); | ||||
|         minutes.value = String(Number(minutes.value) + Number(quotient)).padStart(2, "0"); | ||||
|             minutes.value = String(Number(minutes.value) + Number(quotient)).padStart( | ||||
|                 2, | ||||
|                 "0" | ||||
|             ); | ||||
|         } | ||||
|         if (minutes.value >= 60) { | ||||
|             let quotient = Math.floor(minutes.value / 60); | ||||
| @@ -42,12 +54,6 @@ function update_interval(element) { | ||||
|             minutes.value = String(remainder).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,14 +60,15 @@ function update_select(sel) { | ||||
|         sel.closest("div.reminderContent").querySelector("img.discord-avatar").src = | ||||
|             sel.selectedOptions[0].dataset["webhookAvatar"]; | ||||
|     } 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"]) { | ||||
|         sel.closest("div.reminderContent").querySelector("input.discord-username").value = | ||||
|             sel.selectedOptions[0].dataset["webhookName"]; | ||||
|     } else { | ||||
|         sel.closest("div.reminderContent").querySelector("input.discord-username").value = | ||||
|             ""; | ||||
|             "Reminder"; | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -319,6 +320,7 @@ async function serialize_reminder(node, mode) { | ||||
|         embed_fields: fields, | ||||
|         expires: expiration_time, | ||||
|         interval_seconds: mode !== "template" ? interval.seconds : null, | ||||
|         interval_days: mode !== "template" ? interval.days : null, | ||||
|         interval_months: mode !== "template" ? interval.months : null, | ||||
|         name: node.querySelector('input[name="name"]').value, | ||||
|         tts: node.querySelector('input[name="tts"]').checked, | ||||
| @@ -331,6 +333,9 @@ function deserialize_reminder(reminder, frame, mode) { | ||||
|     // populate channels | ||||
|     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 | ||||
|     for (let prop in reminder) { | ||||
|         if (reminder.hasOwnProperty(prop) && reminder[prop] !== null) { | ||||
| @@ -351,6 +356,8 @@ function deserialize_reminder(reminder, frame, mode) { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     update_interval(frame); | ||||
|  | ||||
|     const lastChild = frame.querySelector("div.embed-multifield-box .embed-field-box"); | ||||
|  | ||||
|     for (let field of reminder["embed_fields"]) { | ||||
| @@ -497,6 +504,8 @@ document.addEventListener("remindersLoaded", (event) => { | ||||
|                 .then((response) => response.json()) | ||||
|                 .then((data) => { | ||||
|                     for (let error of data.errors) show_error(error); | ||||
|  | ||||
|                     deserialize_reminder(data.reminder, node, "reload"); | ||||
|                 }); | ||||
|  | ||||
|             $saveBtn.querySelector("span.icon > i").classList = ["fas fa-check"]; | ||||
| @@ -715,6 +724,7 @@ $createReminderBtn.addEventListener("click", async () => { | ||||
|     let reminder = await serialize_reminder($createReminder, "create"); | ||||
|     if (reminder.error) { | ||||
|         show_error(reminder.error); | ||||
|         $createReminderBtn.querySelector("span.icon > i").classList = ["fas fa-sparkles"]; | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| @@ -832,13 +842,6 @@ $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; | ||||
| const $urlModal = document.querySelector("div#addImageModal"); | ||||
| const $urlInput = $urlModal.querySelector("input"); | ||||
| @@ -894,6 +897,13 @@ document.addEventListener("remindersLoaded", () => { | ||||
|                 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() { | ||||
|   | ||||
| @@ -191,19 +191,8 @@ | ||||
|                     </label> | ||||
|                 </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> | ||||
|             <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"> | ||||
|                     Please first read the <a href="/help/iemanager">support page</a> | ||||
|                 </div> | ||||
|   | ||||
| @@ -27,7 +27,7 @@ | ||||
|             </div> | ||||
|             <div class="tile is-parent"> | ||||
|                 <article class="tile is-child notification"> | ||||
|                     <p class="title">Creating reminders</p> | ||||
|                     <p class="title">Create reminders</p> | ||||
|                     <p class="subtitle">Learn to create reminders for your server</p> | ||||
|                     <div class="content has-text-centered"> | ||||
|                         <a class="button is-size-4 is-rounded is-light" href="/help/create_reminder"> | ||||
| @@ -52,47 +52,47 @@ | ||||
|                 </article> | ||||
|             </div> | ||||
|         </div> | ||||
|         <div class="tile is-ancestor"> | ||||
|             <div class="tile is-parent"> | ||||
|                 <article class="tile is-child notification"> | ||||
|                     <p class="title">Timers</p> | ||||
|                     <p class="subtitle">Learn to manage timers</p> | ||||
|                     <div class="content has-text-centered"> | ||||
|                         <a class="button is-size-4 is-rounded is-light" href="/help/timers"> | ||||
|                             <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"> | ||||
|                 <article class="tile is-child notification"> | ||||
|                     <p class="title">Todo Lists</p> | ||||
|                     <p class="subtitle">Learn to manage various todo lists</p> | ||||
|                     <div class="content has-text-centered"> | ||||
|                         <a class="button is-size-4 is-rounded is-light" href="/help/todo_lists"> | ||||
|                             <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"> | ||||
|                 <article class="tile is-child notification"> | ||||
|                     <p class="title">Macros</p> | ||||
|                     <p class="subtitle">Learn how to create combination commands called macros, to suit advanced use cases</p> | ||||
|                     <div class="content has-text-centered"> | ||||
|                         <a class="button is-size-4 is-rounded is-light" href="/help/macros"> | ||||
|                             <p class="is-size-4"> | ||||
|                                 Read <span class="icon"><i class="fas fa-chevron-right"></i></span> | ||||
|                             </p> | ||||
|                         </a> | ||||
|                     </div> | ||||
|                 </article> | ||||
|             </div> | ||||
|         </div> | ||||
| <!--        <div class="tile is-ancestor">--> | ||||
| <!--            <div class="tile is-parent">--> | ||||
| <!--                <article class="tile is-child notification">--> | ||||
| <!--                    <p class="title">Timers</p>--> | ||||
| <!--                    <p class="subtitle">Learn to manage timers</p>--> | ||||
| <!--                    <div class="content has-text-centered">--> | ||||
| <!--                        <a class="button is-size-4 is-rounded is-light" href="/help/timers">--> | ||||
| <!--                            <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">--> | ||||
| <!--                <article class="tile is-child notification">--> | ||||
| <!--                    <p class="title">Todo Lists</p>--> | ||||
| <!--                    <p class="subtitle">Learn to manage various todo lists</p>--> | ||||
| <!--                    <div class="content has-text-centered">--> | ||||
| <!--                        <a class="button is-size-4 is-rounded is-light" href="/help/todo_lists">--> | ||||
| <!--                            <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">--> | ||||
| <!--                <article class="tile is-child notification">--> | ||||
| <!--                    <p class="title">Macros</p>--> | ||||
| <!--                    <p class="subtitle">Learn how to create combination commands called macros, to suit advanced use cases</p>--> | ||||
| <!--                    <div class="content has-text-centered">--> | ||||
| <!--                        <a class="button is-size-4 is-rounded is-light" href="/help/macros">--> | ||||
| <!--                            <p class="is-size-4">--> | ||||
| <!--                                Read <span class="icon"><i class="fas fa-chevron-right"></i></span>--> | ||||
| <!--                            </p>--> | ||||
| <!--                        </a>--> | ||||
| <!--                    </div>--> | ||||
| <!--                </article>--> | ||||
| <!--            </div>--> | ||||
| <!--        </div>--> | ||||
|         <div class="tile is-ancestor"> | ||||
|             <div class="tile is-parent"> | ||||
|                 <article class="tile is-child notification"> | ||||
| @@ -107,19 +107,6 @@ | ||||
|                     </div> | ||||
|                 </article> | ||||
|             </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"> | ||||
|                 <article class="tile is-child notification"> | ||||
|                     <p class="title">Import/Export</p> | ||||
| @@ -133,6 +120,19 @@ | ||||
|                     </div> | ||||
|                 </article> | ||||
|             </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> | ||||
|  | ||||
|   | ||||
| @@ -28,7 +28,10 @@ | ||||
|             <div class="container"> | ||||
|                 <p class="title">Create reminders via the dashboard</p> | ||||
|                 <p class="content"> | ||||
|                     Reminders can also be created on the dashboard. | ||||
|                     Reminders can also be created on the dashboard. The dashboard offers more options for configuring | ||||
|                     reminders, and offers templates for quick recreation of reminders. | ||||
|  | ||||
|                     <a href="/dashboard">Access the dashboard.</a> | ||||
|                 </p> | ||||
|             </div> | ||||
|         </div> | ||||
|   | ||||
| @@ -12,7 +12,7 @@ | ||||
|     <section class="hero is-small"> | ||||
|         <div class="hero-body"> | ||||
|             <div class="container"> | ||||
|                 <p class="title">Export your data</p> | ||||
|                 <p class="title">Export data</p> | ||||
|                 <p class="content"> | ||||
|                     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. | ||||
| @@ -26,8 +26,7 @@ | ||||
|             <div class="container"> | ||||
|                 <p class="title">Import data</p> | ||||
|                 <p class="content"> | ||||
|                     You can import previous exports or modified exports. When importing a file, <strong>existing data | ||||
|                     will be overwritten</strong>. | ||||
|                     You can import previous exports or modified exports. When importing a file, the new data will be added alongside existing data. | ||||
|                 </p> | ||||
|             </div> | ||||
|         </div> | ||||
| @@ -55,7 +54,7 @@ | ||||
|                         </figure> | ||||
|                     </li> | ||||
|                     <li> | ||||
|                         Make edits to the spreadsheet. You can add, edit, and remove rows for reminders. Don't remove the title row. | ||||
|                         Make edits to the spreadsheet. You can add, edit, and remove rows for reminders. Don't remove the top-most (title) row. | ||||
|                         <figure> | ||||
|                             <img src="/static/img/support/iemanager/edit_spreadsheet.png" alt="Editing spreadsheet"> | ||||
|                         </figure> | ||||
| @@ -70,7 +69,7 @@ | ||||
|                 Other spreadsheet tools can also be used to edit exports, as long as they are properly configured: | ||||
|                 <ul> | ||||
|                     <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: | ||||
|                         <figure> | ||||
|                             <img src="/static/img/support/iemanager/sheets_settings.png" alt="Google sheets import settings"> | ||||
|   | ||||
| @@ -49,7 +49,7 @@ | ||||
|                 <p class="content"> | ||||
|                     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 | ||||
|                     for marking certain dates. | ||||
|                     for marking calendar events. | ||||
|                 </p> | ||||
|             </div> | ||||
|         </div> | ||||
| @@ -61,7 +61,8 @@ | ||||
|                 <p class="title">Interval expiration</p> | ||||
|                 <p class="content"> | ||||
|                     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. | ||||
|                     This is optional, and if omitted, the reminder will repeat indefinitely. Otherwise, the reminder | ||||
|                     will be deleted once the expiration date is reached. | ||||
|                 </p> | ||||
|             </div> | ||||
|         </div> | ||||
|   | ||||
		Reference in New Issue
	
	Block a user