Compare commits

...

5 commits

Author SHA1 Message Date
b374c6ed66 Update for Nix module 2025-03-26 10:39:47 -07:00
677a97d5cc Fix leaderboard truncation 2025-03-25 19:23:31 -07:00
eccd55542d Update 2025-03-25 19:22:24 -07:00
665d5c025b useFetchCargoVendor = true; 2025-03-25 19:02:21 -07:00
b3bbe0f59e flake.lock: Update
Flake lock file updates:

• Updated input 'nixpkgs':
    'github:NixOS/nixpkgs/18324978d632ffc55ef1d928e81630c620f4f447?narHash=sha256-1SvMQm2DwofNxXVtNWWtIcTh7GctEVrS/Xel/mdc6iY%3D' (2023-08-24)
  → 'github:NixOS/nixpkgs/1e5b653dff12029333a6546c11e108ede13052eb?narHash=sha256-G5n%2BFOXLXcRx%2B3hCJ6Rt6ZQyF1zqQ0DL0sWAMn2Nk0w%3D' (2025-03-22)
• Updated input 'rust-overlay':
    'github:oxalica/rust-overlay/673e2d3d2a3951adc6f5e3351c9fce6ad130baed?narHash=sha256-zdN6UVtEml7t0WQHVy0avsE%2BTWJLklXnqJyiPaOa8u0%3D' (2023-08-25)
  → 'github:oxalica/rust-overlay/b4c18f262dbebecb855136c1ed8047b99a9c75b6?narHash=sha256-eQnw8ufyLmrboODU8RKVNh2Mv7SACzdoFrRUV5zdNNE%3D' (2025-03-25)
• Removed input 'rust-overlay/flake-utils'
• Removed input 'rust-overlay/flake-utils/systems'
• Updated input 'rust-overlay/nixpkgs':
    'github:NixOS/nixpkgs/96ba1c52e54e74c3197f4d43026b3f3d92e83ff9?narHash=sha256-eKyxW4OohHQx9Urxi7TQlFBTDWII%2BF%2Bx2hklDOQPB50%3D' (2023-04-13)
  → 'github:NixOS/nixpkgs/4bc9c909d9ac828a039f288cf872d16d38185db8?narHash=sha256-nIYdTAiKIGnFNugbomgBJR%2BXv5F1ZQU%2BHfaBqJKroC0%3D' (2025-01-08)
2025-03-25 18:58:52 -07:00
9 changed files with 1575 additions and 590 deletions

1923
Cargo.lock generated

File diff suppressed because it is too large Load diff

View file

@ -1,21 +1,28 @@
[package] [package]
name = "ji-chan" name = "ji-chan"
version = "0.1.0" version = "0.1.0"
edition = "2021" authors = ["ElnuDev <elnu@elnu.com>"]
edition = "2024"
description = "Ji-chan (字ちゃん), the Discord bot and website manager for Tegaki Tuesday, the weekly Japanese handwriting challenge. https://tegakituesday.com/"
homepage = "https://tegakituesday.com/"
repository = "https://git.elnu.com/tegakituesday/ji-chan"
license = "GPL-3.0"
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
[dependencies] [dependencies]
dotenv = "0.15" dotenv = "0.15"
rand = "0.8" rand = "0.9"
tracing = "0.1" tracing = "0.1"
tracing-subscriber = "0.3" tracing-subscriber = "0.3"
serde_json = "1.0" serde_json = "1.0"
reqwest = "0.11" reqwest = "0.12"
slug = "0.1" slug = "0.1"
unicode_hfwidth = "0.2" unicode_hfwidth = "0.2"
fs_extra = "1.3" fs_extra = "1.3"
poise = "0.5.5" poise = "0.5.7"
clap = { version = "4.5.32", features = ["derive"] }
lazy_static = "1.5.0"
[dependencies.tokio] [dependencies.tokio]
version = "1.32" version = "1.32"

52
flake.lock generated
View file

@ -1,30 +1,12 @@
{ {
"nodes": { "nodes": {
"flake-utils": {
"inputs": {
"systems": "systems"
},
"locked": {
"lastModified": 1681202837,
"narHash": "sha256-H+Rh19JDwRtpVPAWp64F+rlEtxUWBAQW28eAi3SRSzg=",
"owner": "numtide",
"repo": "flake-utils",
"rev": "cfacdce06f30d2b68473a46042957675eebb3401",
"type": "github"
},
"original": {
"owner": "numtide",
"repo": "flake-utils",
"type": "github"
}
},
"nixpkgs": { "nixpkgs": {
"locked": { "locked": {
"lastModified": 1692913444, "lastModified": 1742669843,
"narHash": "sha256-1SvMQm2DwofNxXVtNWWtIcTh7GctEVrS/Xel/mdc6iY=", "narHash": "sha256-G5n+FOXLXcRx+3hCJ6Rt6ZQyF1zqQ0DL0sWAMn2Nk0w=",
"owner": "NixOS", "owner": "NixOS",
"repo": "nixpkgs", "repo": "nixpkgs",
"rev": "18324978d632ffc55ef1d928e81630c620f4f447", "rev": "1e5b653dff12029333a6546c11e108ede13052eb",
"type": "github" "type": "github"
}, },
"original": { "original": {
@ -36,11 +18,11 @@
}, },
"nixpkgs_2": { "nixpkgs_2": {
"locked": { "locked": {
"lastModified": 1681358109, "lastModified": 1736320768,
"narHash": "sha256-eKyxW4OohHQx9Urxi7TQlFBTDWII+F+x2hklDOQPB50=", "narHash": "sha256-nIYdTAiKIGnFNugbomgBJR+Xv5F1ZQU+HfaBqJKroC0=",
"owner": "NixOS", "owner": "NixOS",
"repo": "nixpkgs", "repo": "nixpkgs",
"rev": "96ba1c52e54e74c3197f4d43026b3f3d92e83ff9", "rev": "4bc9c909d9ac828a039f288cf872d16d38185db8",
"type": "github" "type": "github"
}, },
"original": { "original": {
@ -58,15 +40,14 @@
}, },
"rust-overlay": { "rust-overlay": {
"inputs": { "inputs": {
"flake-utils": "flake-utils",
"nixpkgs": "nixpkgs_2" "nixpkgs": "nixpkgs_2"
}, },
"locked": { "locked": {
"lastModified": 1692929460, "lastModified": 1742870002,
"narHash": "sha256-zdN6UVtEml7t0WQHVy0avsE+TWJLklXnqJyiPaOa8u0=", "narHash": "sha256-eQnw8ufyLmrboODU8RKVNh2Mv7SACzdoFrRUV5zdNNE=",
"owner": "oxalica", "owner": "oxalica",
"repo": "rust-overlay", "repo": "rust-overlay",
"rev": "673e2d3d2a3951adc6f5e3351c9fce6ad130baed", "rev": "b4c18f262dbebecb855136c1ed8047b99a9c75b6",
"type": "github" "type": "github"
}, },
"original": { "original": {
@ -74,21 +55,6 @@
"repo": "rust-overlay", "repo": "rust-overlay",
"type": "github" "type": "github"
} }
},
"systems": {
"locked": {
"lastModified": 1681028828,
"narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=",
"owner": "nix-systems",
"repo": "default",
"rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e",
"type": "github"
},
"original": {
"owner": "nix-systems",
"repo": "default",
"type": "github"
}
} }
}, },
"root": "root", "root": "root",

View file

@ -40,6 +40,8 @@ Some utility commands:
version = "0.1.0"; version = "0.1.0";
nativeBuildInputs = with pkgs; [ pkg-config ]; nativeBuildInputs = with pkgs; [ pkg-config ];
buildInputs = with pkgs; [ openssl ]; buildInputs = with pkgs; [ openssl ];
# rustPlatform.fetchCargoTarball is deprecated in favor of rustPlatform.fetchCargoVendor.
useFetchCargoVendor = true;
cargoHash = "sha256-eKsQsOIiZDpDOGLMcpU+dL/e/UBEnbQIgcVPKi5xpC8="; cargoHash = "sha256-eKsQsOIiZDpDOGLMcpU+dL/e/UBEnbQIgcVPKi5xpC8=";
meta = with nixpkgs.lib; { meta = with nixpkgs.lib; {
homepage = "https://tegakituesday.com"; homepage = "https://tegakituesday.com";
@ -50,5 +52,71 @@ Some utility commands:
}; };
}; };
}; };
nixosModules.default = { config, ... }: let
lib = nixpkgs.lib;
in {
options.services.ji-chan = {
enable = lib.mkEnableOption (lib.mdDoc "ji-chan service");
package = lib.mkOption {
type = lib.types.package;
default = self.packages.${system}.ji-chan;
defaultText = "pkgs.ji-chan";
description = lib.mdDoc ''
The ji-chan package that should be used.
'';
};
domain = lib.mkOption {
type = lib.types.str;
default = "tegakituesday.com";
description = lib.mdDoc ''
Website domain
'';
};
token = lib.mkOption {
type = lib.types.str;
description = lib.mdDoc ''
Discord token
'';
};
prefix = lib.mkOption {
type = lib.types.str;
description = lib.mdDoc ''
Traditional text command prefix, including space, if any
'';
};
hugo = lib.mkOption {
type = lib.types.str;
description = lib.mdDoc ''
Path to Hugo project where site rebuilds
'';
};
guildData = lib.mkOption {
type = lib.types.str;
description = lib.mdDoc ''
Path to guild data JSON file
'';
};
};
config.systemd.services.ji-chan = let
cfg = config.services.ji-chan;
pkg = self.packages.${system}.ji-chan;
in lib.mkIf cfg.enable {
description = pkg.meta.description;
after = [ "network.target" ];
wantedBy = [ "network.target" ];
serviceConfig = {
ExecStart = ''
${cfg.package}/bin/ji-chan \
--domain ${cfg.domain} \
--token ${cfg.token} \
--prefix ${cfg.prefix} \
--hugo ${cfg.hugo} \
--guildData ${cfg.guildData}
'';
Restart = "always";
DynamicUser = true;
};
};
};
}; };
} }

View file

@ -1,4 +1,5 @@
use crate::utils::*; use crate::utils::*;
use crate::ARGS;
use serde_json::Map; use serde_json::Map;
use crate::serenity; use crate::serenity;
@ -21,7 +22,7 @@ use std::path::Path;
pub async fn challenge(ctx: Context<'_>) -> Result<(), Error> { pub async fn challenge(ctx: Context<'_>) -> Result<(), Error> {
ctx.say(format!( ctx.say(format!(
"Tegaki Tuesday #{n}: <https://{domain}/{n}>", "Tegaki Tuesday #{n}: <https://{domain}/{n}>",
domain = get_domain(), domain = ARGS.domain,
n = get_challenge_number() n = get_challenge_number()
)) ))
.await?; .await?;
@ -199,7 +200,7 @@ pub async fn submit(ctx: Context<'_>, submission: serenity::Attachment) -> Resul
let thank_you = &format!( let thank_you = &format!(
"Thank you for submitting! You can view your submission at <https://{domain}/{}>", "Thank you for submitting! You can view your submission at <https://{domain}/{}>",
challenge_number, challenge_number,
domain = get_domain() domain = ARGS.domain
); );
let mut repost_here = true; let mut repost_here = true;
match ctx { match ctx {
@ -262,7 +263,7 @@ pub async fn submit(ctx: Context<'_>, submission: serenity::Attachment) -> Resul
channel channel
.send_message(&ctx.serenity_context().http, |m| { .send_message(&ctx.serenity_context().http, |m| {
m.embed(|e| { m.embed(|e| {
let domain = get_domain(); let domain = &ARGS.domain;
let n = get_challenge_number(); let n = get_challenge_number();
let mut description = format!( let mut description = format!(
"New submission to [Tegaki Tuesday #{n}](https://{domain}/{n})!" "New submission to [Tegaki Tuesday #{n}](https://{domain}/{n})!"
@ -344,7 +345,7 @@ pub async fn images(ctx: Context<'_>) -> Result<(), Error> {
to_fullwidth(&(i + 1).to_string()), to_fullwidth(&(i + 1).to_string()),
challenge_number, challenge_number,
image, image,
domain = get_domain() domain = ARGS.domain
)); ));
} }
ctx.say(message).await?; ctx.say(message).await?;

View file

@ -1,10 +1,8 @@
use crate::utils::get_domain;
use crate::Context; use crate::Context;
use crate::Error; use crate::Error;
use crate::ARGS;
use poise::command; use poise::command;
use std::env;
// TODO: Implement proper help text for command-specific help, // TODO: Implement proper help text for command-specific help,
// see https://github.com/kangalioo/poise/blob/90ac24a8ef621ec6dc3fc452762dc9cfa144f693/examples/framework_usage/main.rs#L18-L38 // see https://github.com/kangalioo/poise/blob/90ac24a8ef621ec6dc3fc452762dc9cfa144f693/examples/framework_usage/main.rs#L18-L38
#[command( #[command(
@ -33,8 +31,8 @@ __**Kanji 漢字**__
:game_die: `{p}kyoiku <grade|all>` Random Kyōiku kanji :game_die: `{p}kyoiku <grade|all>` Random Kyōiku kanji
:game_die: `{p}jlpt <level|all>` Random JLPT kanji :game_die: `{p}jlpt <level|all>` Random JLPT kanji
:game_die: `{p}hyogai <group|all>` Random Hyōgai kanji", :game_die: `{p}hyogai <group|all>` Random Hyōgai kanji",
p = env::var("PREFIX").unwrap(), p = ARGS.prefix,
domain = get_domain(), domain = ARGS.domain,
); );
ctx.say(message).await?; ctx.say(message).await?;
Ok(()) Ok(())

View file

@ -2,10 +2,10 @@
use crate::Context; use crate::Context;
use crate::Error; use crate::Error;
use crate::ARGS;
use poise::command; use poise::command;
use serde_json::json; use serde_json::json;
use std::env;
use crate::utils::*; use crate::utils::*;
@ -100,7 +100,7 @@ pub async fn announcechallenge(ctx: Context<'_>) -> Result<(), Error> {
let message = format!("Welcome to the **{n}{th}** weekly **Tegaki Tuesday** (手書きの火曜日) handwriting challenge! :pen_fountain: The prompt is available in both Japanese and English on the website at <https://{domain}/{n}>. let message = format!("Welcome to the **{n}{th}** weekly **Tegaki Tuesday** (手書きの火曜日) handwriting challenge! :pen_fountain: The prompt is available in both Japanese and English on the website at <https://{domain}/{n}>.
You can make submissions in both languages, but please submit in your target language first. Submissions can be submitted by uploading the image to this channel along with the `{p}submit` command. By submitting, you agree to having your work posted to the website under the Attribution-ShareAlike 4.0 Unported (CC BY-SA 4.0) license, attributed to your Discord account. (<https://creativecommons.org/licenses/by-sa/4.0>).", You can make submissions in both languages, but please submit in your target language first. Submissions can be submitted by uploading the image to this channel along with the `{p}submit` command. By submitting, you agree to having your work posted to the website under the Attribution-ShareAlike 4.0 Unported (CC BY-SA 4.0) license, attributed to your Discord account. (<https://creativecommons.org/licenses/by-sa/4.0>).",
domain = get_domain(), domain = ARGS.domain,
n = challenge_number, n = challenge_number,
th = match challenge_number % 10 { th = match challenge_number % 10 {
1 => "ˢᵗ", 1 => "ˢᵗ",
@ -108,7 +108,7 @@ You can make submissions in both languages, but please submit in your target lan
3 => "ʳᵈ", 3 => "ʳᵈ",
_ => "ᵗʰ" _ => "ᵗʰ"
}, },
p = env::var("PREFIX").unwrap() p = ARGS.prefix
); );
send(ctx, &message, true, false).await?; send(ctx, &message, true, false).await?;
ctx.say("Announced!").await?; ctx.say("Announced!").await?;

View file

@ -1,7 +1,8 @@
mod commands; mod commands;
mod utils; mod utils;
use std::env; use std::env::{self, VarError};
use lazy_static::lazy_static;
type Error = Box<dyn std::error::Error + Send + Sync>; type Error = Box<dyn std::error::Error + Send + Sync>;
type Context<'a> = poise::Context<'a, Data, Error>; type Context<'a> = poise::Context<'a, Data, Error>;
@ -12,6 +13,7 @@ pub struct Data {}
use commands::{challenge::*, kanji::*, meta::*, owner::*}; use commands::{challenge::*, kanji::*, meta::*, owner::*};
use poise::serenity_prelude as serenity; use poise::serenity_prelude as serenity;
use poise::serenity_prelude::model::gateway::GatewayIntents; use poise::serenity_prelude::model::gateway::GatewayIntents;
use clap::Parser;
#[poise::command(prefix_command)] #[poise::command(prefix_command)]
async fn register(ctx: Context<'_>) -> Result<(), Error> { async fn register(ctx: Context<'_>) -> Result<(), Error> {
@ -19,12 +21,50 @@ async fn register(ctx: Context<'_>) -> Result<(), Error> {
Ok(()) Ok(())
} }
#[tokio::main] #[derive(Parser)]
async fn main() { #[clap(author, version, about)]
struct Arguments {
#[clap(long, help = "Website domain", default_value = "tegakituesday.com")]
pub domain: String,
#[clap(long = "token", help = "Discord token")]
pub discord_token: String,
#[clap(long, help = "Traditional text command prefix, including space, if any", default_value = "-h")]
pub prefix: String,
#[clap(long, help = "Path to Hugo project where site rebuilds")]
pub hugo: String,
#[clap(long = "guilds", help = "Guild data JSON file", default_value = "guilds.json")]
pub guild_data: String,
}
impl Arguments {
fn from_dotenv() -> Result<Self, VarError> {
Ok(Self {
domain: env::var("DOMAIN")?,
discord_token: env::var("DISCORD_TOKEN")?,
prefix: env::var("PREFIX")?,
hugo: env::var("HUGO")?.into(),
guild_data: env::var("GUILD_DATA")?.into(),
})
}
}
lazy_static! {
static ref ARGS: Arguments = {
match env::args().count() {
1 => {
// This will load the environment variables located at `./.env`, relative to // This will load the environment variables located at `./.env`, relative to
// the CWD. See `./.env.example` for an example on how to structure this. // the CWD. See `./.env.example` for an example on how to structure this.
dotenv::dotenv().expect("Failed to load .env file"); dotenv::dotenv().expect("Failed to load .env file");
Arguments::from_dotenv().unwrap()
},
_ => Arguments::parse(),
}
};
}
#[tokio::main]
async fn main() {
// Initialize the logger to use environment variables. // Initialize the logger to use environment variables.
// //
// In this case, a good default is setting the environment variable // In this case, a good default is setting the environment variable
@ -59,12 +99,12 @@ async fn main() {
so(), so(),
], ],
prefix_options: poise::PrefixFrameworkOptions { prefix_options: poise::PrefixFrameworkOptions {
prefix: Some(env::var("PREFIX").expect("Expected a prefix in the environment")), prefix: Some(ARGS.prefix.clone()),
..Default::default() ..Default::default()
}, },
..Default::default() ..Default::default()
}) })
.token(std::env::var("DISCORD_TOKEN").expect("Expected a token in the environment")) .token(&ARGS.discord_token)
.intents( .intents(
GatewayIntents::GUILD_MESSAGES GatewayIntents::GUILD_MESSAGES
| GatewayIntents::DIRECT_MESSAGES | GatewayIntents::DIRECT_MESSAGES

View file

@ -1,6 +1,7 @@
use crate::serenity; use crate::serenity;
use crate::Context; use crate::Context;
use crate::Error; use crate::Error;
use crate::ARGS;
use rand::seq::IteratorRandom; use rand::seq::IteratorRandom;
use serde_json::Map; use serde_json::Map;
use serde_json::Value; use serde_json::Value;
@ -52,7 +53,7 @@ mod tests {
} }
pub fn get_challenge_number() -> i32 { pub fn get_challenge_number() -> i32 {
let challenge_dir = format!("{}/content/challenges", env::var("HUGO").unwrap()); let challenge_dir = format!("{}/content/challenges", ARGS.hugo);
let paths = fs::read_dir(challenge_dir).unwrap(); let paths = fs::read_dir(challenge_dir).unwrap();
let mut max = 0; let mut max = 0;
for path in paths { for path in paths {
@ -72,20 +73,12 @@ pub fn get_challenge_number() -> i32 {
max max
} }
pub fn get_domain() -> String {
env::var("DOMAIN").unwrap()
}
pub fn get_hugo_path() -> String {
env::var("HUGO").unwrap()
}
pub fn get_submission_images_dir() -> String { pub fn get_submission_images_dir() -> String {
format!("{}/assets/{}", get_hugo_path(), get_challenge_number()) format!("{}/assets/{}", ARGS.hugo, get_challenge_number())
} }
pub fn get_submission_data_path(challenge: i32) -> String { pub fn get_submission_data_path(challenge: i32) -> String {
format!("{}/data/challenges/{}.json", get_hugo_path(), challenge) format!("{}/data/challenges/{}.json", ARGS.hugo, challenge)
} }
pub fn get_current_submission_data_path() -> String { pub fn get_current_submission_data_path() -> String {
@ -153,17 +146,13 @@ pub fn to_fullwidth(string: &str) -> String {
pub fn rebuild_site() { pub fn rebuild_site() {
Command::new("./build.sh") Command::new("./build.sh")
.current_dir(get_hugo_path()) .current_dir(&ARGS.hugo)
.status() .status()
.expect("Failed to rebuild site"); .expect("Failed to rebuild site");
} }
pub fn get_guild_data_path() -> String {
env::var("GUILD_DATA").unwrap()
}
pub fn get_guild_data() -> Map<String, Value> { pub fn get_guild_data() -> Map<String, Value> {
let guild_data_path = get_guild_data_path(); let guild_data_path = &ARGS.guild_data;
let guild_data_json = match File::open(&guild_data_path) { let guild_data_json = match File::open(&guild_data_path) {
Ok(mut file) => { Ok(mut file) => {
let mut json = String::new(); let mut json = String::new();
@ -187,7 +176,7 @@ pub fn set_guild_data(guild_data: Map<String, Value>) {
let mut guild_data_file = OpenOptions::new() let mut guild_data_file = OpenOptions::new()
.write(true) .write(true)
.truncate(true) .truncate(true)
.open(get_guild_data_path()) .open(&ARGS.guild_data)
.unwrap(); .unwrap();
guild_data_file guild_data_file
.write_all( .write_all(
@ -248,7 +237,7 @@ pub async fn send(ctx: Context<'_>, message: &str, ping: bool, pin: bool) -> Res
} }
pub fn random_from_string(string: &str) -> char { pub fn random_from_string(string: &str) -> char {
string.chars().choose(&mut rand::thread_rng()).unwrap() string.chars().choose(&mut rand::rng()).unwrap()
} }
pub fn get_so_diagram(kanji: char) -> String { pub fn get_so_diagram(kanji: char) -> String {
@ -368,7 +357,7 @@ pub async fn random_kanji(
if subcategory == "ALL" { if subcategory == "ALL" {
let subcategory_key = subcategories let subcategory_key = subcategories
.keys() .keys()
.choose(&mut rand::thread_rng()) .choose(&mut rand::rng())
.unwrap(); .unwrap();
let list = subcategories[subcategory_key].as_str().unwrap(); let list = subcategories[subcategory_key].as_str().unwrap();
let kanji = random_from_string(list); let kanji = random_from_string(list);
@ -440,6 +429,7 @@ pub async fn leaderboard(ctx: &Context<'_>) -> Result<(), Error> {
let mut file = std::fs::OpenOptions::new() let mut file = std::fs::OpenOptions::new()
.create(true) .create(true)
.write(true) .write(true)
.truncate(true)
.open(env::var("LEADERBOARD").unwrap()) .open(env::var("LEADERBOARD").unwrap())
.unwrap(); .unwrap();
file.write_all(leaderboard_html.as_bytes())?; file.write_all(leaderboard_html.as_bytes())?;