From 7a06c4cea7d05d056277bf3e313e3cb5c9eb7fa2 Mon Sep 17 00:00:00 2001 From: Daan Boerlage Date: Sat, 27 Aug 2022 20:15:31 +0200 Subject: [PATCH] Initial Commit --- .gitignore | 1 + Cargo.lock | 747 +++++++++++++++++++++++++++++++++ Cargo.toml | 19 + readme.md | 14 + src/args.rs | 14 + src/main.rs | 120 ++++++ src/rules/has_dash_in_title.rs | 20 + src/rules/has_episode_name.rs | 18 + src/rules/has_fluff.rs | 24 ++ src/rules/missing_separator.rs | 18 + src/rules/mod.rs | 48 +++ src/utils/constructors.rs | 4 + src/utils/mod.rs | 2 + src/utils/status_icon.rs | 15 + 14 files changed, 1064 insertions(+) create mode 100644 .gitignore create mode 100644 Cargo.lock create mode 100644 Cargo.toml create mode 100644 readme.md create mode 100644 src/args.rs create mode 100644 src/main.rs create mode 100644 src/rules/has_dash_in_title.rs create mode 100644 src/rules/has_episode_name.rs create mode 100644 src/rules/has_fluff.rs create mode 100644 src/rules/missing_separator.rs create mode 100644 src/rules/mod.rs create mode 100644 src/utils/constructors.rs create mode 100644 src/utils/mod.rs create mode 100644 src/utils/status_icon.rs diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..ea8c4bf --- /dev/null +++ b/.gitignore @@ -0,0 +1 @@ +/target diff --git a/Cargo.lock b/Cargo.lock new file mode 100644 index 0000000..c3f3885 --- /dev/null +++ b/Cargo.lock @@ -0,0 +1,747 @@ +# This file is automatically @generated by Cargo. +# It is not intended for manual editing. +version = 3 + +[[package]] +name = "aho-corasick" +version = "0.7.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e37cfd5e7657ada45f742d6e99ca5788580b5c529dc78faf11ece6dc702656f" +dependencies = [ + "memchr", +] + +[[package]] +name = "atty" +version = "0.2.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d9b39be18770d11421cdb1b9947a45dd3f37e93092cbf377614828a319d5fee8" +dependencies = [ + "hermit-abi", + "libc", + "winapi", +] + +[[package]] +name = "autocfg" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" + +[[package]] +name = "bitflags" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" + +[[package]] +name = "bytes" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec8a7b6a70fde80372154c65702f00a0f56f3e1c36abbc6c440484be248856db" + +[[package]] +name = "cfg-if" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" + +[[package]] +name = "clap" +version = "3.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "29e724a68d9319343bb3328c9cc2dfde263f4b3142ee1059a9980580171c954b" +dependencies = [ + "atty", + "bitflags", + "clap_derive", + "clap_lex", + "indexmap", + "once_cell", + "strsim", + "termcolor", + "textwrap", +] + +[[package]] +name = "clap_derive" +version = "3.2.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "13547f7012c01ab4a0e8f8967730ada8f9fdf419e8b6c792788f39cf4e46eefa" +dependencies = [ + "heck", + "proc-macro-error", + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "clap_lex" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2850f2f5a82cbf437dd5af4d49848fbdfc27c157c3d010345776f952765261c5" +dependencies = [ + "os_str_bytes", +] + +[[package]] +name = "dirs" +version = "4.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca3aa72a6f96ea37bbc5aa912f6788242832f75369bdfdadcb0e38423f100059" +dependencies = [ + "dirs-sys", +] + +[[package]] +name = "dirs-sys" +version = "0.3.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1b1d1d91c932ef41c0f2663aa8b0ca0342d444d842c06914aa0a7e352d0bada6" +dependencies = [ + "libc", + "redox_users", + "winapi", +] + +[[package]] +name = "eyre" +version = "0.6.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4c2b6b5a29c02cdc822728b7d7b8ae1bab3e3b05d44522770ddd49722eeac7eb" +dependencies = [ + "indenter", + "once_cell", +] + +[[package]] +name = "futures" +version = "0.3.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ab30e97ab6aacfe635fad58f22c2bb06c8b685f7421eb1e064a729e2a5f481fa" +dependencies = [ + "futures-channel", + "futures-core", + "futures-executor", + "futures-io", + "futures-sink", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-channel" +version = "0.3.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2bfc52cbddcfd745bf1740338492bb0bd83d76c67b445f91c5fb29fae29ecaa1" +dependencies = [ + "futures-core", + "futures-sink", +] + +[[package]] +name = "futures-core" +version = "0.3.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d2acedae88d38235936c3922476b10fced7b2b68136f5e3c03c2d5be348a1115" + +[[package]] +name = "futures-executor" +version = "0.3.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d11aa21b5b587a64682c0094c2bdd4df0076c5324961a40cc3abd7f37930528" +dependencies = [ + "futures-core", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-io" +version = "0.3.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93a66fc6d035a26a3ae255a6d2bca35eda63ae4c5512bef54449113f7a1228e5" + +[[package]] +name = "futures-macro" +version = "0.3.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0db9cce532b0eae2ccf2766ab246f114b56b9cf6d445e00c2549fbc100ca045d" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "futures-sink" +version = "0.3.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca0bae1fe9752cf7fd9b0064c674ae63f97b37bc714d745cbde0afb7ec4e6765" + +[[package]] +name = "futures-task" +version = "0.3.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "842fc63b931f4056a24d59de13fb1272134ce261816e063e634ad0c15cdc5306" + +[[package]] +name = "futures-util" +version = "0.3.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f0828a5471e340229c11c77ca80017937ce3c58cb788a17e5f1c2d5c485a9577" +dependencies = [ + "futures-channel", + "futures-core", + "futures-io", + "futures-macro", + "futures-sink", + "futures-task", + "memchr", + "pin-project-lite", + "pin-utils", + "slab", +] + +[[package]] +name = "getrandom" +version = "0.2.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4eb1a864a501629691edf6c15a593b7a51eebaa1e8468e9ddc623de7c9b58ec6" +dependencies = [ + "cfg-if", + "libc", + "wasi", +] + +[[package]] +name = "hashbrown" +version = "0.12.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888" + +[[package]] +name = "heck" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2540771e65fc8cb83cd6e8a237f70c319bd5c29f78ed1084ba5d50eeac86f7f9" + +[[package]] +name = "hermit-abi" +version = "0.1.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33" +dependencies = [ + "libc", +] + +[[package]] +name = "indenter" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce23b50ad8242c51a442f3ff322d56b02f08852c77e4c0b4d3fd684abc89c683" + +[[package]] +name = "indexmap" +version = "1.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "10a35a97730320ffe8e2d410b5d3b69279b98d2c14bdb8b70ea89ecf7888d41e" +dependencies = [ + "autocfg", + "hashbrown", +] + +[[package]] +name = "itoa" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c8af84674fe1f223a982c933a0ee1086ac4d4052aa0fb8060c12c6ad838e754" + +[[package]] +name = "libc" +version = "0.2.132" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8371e4e5341c3a96db127eb2465ac681ced4c433e01dd0e938adbef26ba93ba5" + +[[package]] +name = "lock_api" +version = "0.4.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "327fa5b6a6940e4699ec49a9beae1ea4845c6bab9314e4f84ac68742139d8c53" +dependencies = [ + "autocfg", + "scopeguard", +] + +[[package]] +name = "log" +version = "0.4.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "abb12e687cfb44aa40f41fc3978ef76448f9b6038cad6aef4259d3c095a2382e" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "media-linter" +version = "0.1.0" +dependencies = [ + "clap", + "dirs", + "eyre", + "futures", + "regex", + "serde", + "serde_json", + "tokio", + "toml", + "uuid", + "walkdir", +] + +[[package]] +name = "memchr" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2dffe52ecf27772e601905b7522cb4ef790d2cc203488bbd0e2fe85fcb74566d" + +[[package]] +name = "mio" +version = "0.8.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "57ee1c23c7c63b0c9250c339ffdc69255f110b298b901b9f6c82547b7b87caaf" +dependencies = [ + "libc", + "log", + "wasi", + "windows-sys", +] + +[[package]] +name = "num_cpus" +version = "1.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19e64526ebdee182341572e50e9ad03965aa510cd94427a4549448f285e957a1" +dependencies = [ + "hermit-abi", + "libc", +] + +[[package]] +name = "once_cell" +version = "1.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "074864da206b4973b84eb91683020dbefd6a8c3f0f38e054d93954e891935e4e" + +[[package]] +name = "os_str_bytes" +version = "6.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ff7415e9ae3fff1225851df9e0d9e4e5479f947619774677a63572e55e80eff" + +[[package]] +name = "parking_lot" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3742b2c103b9f06bc9fff0a37ff4912935851bee6d36f3c02bcc755bcfec228f" +dependencies = [ + "lock_api", + "parking_lot_core", +] + +[[package]] +name = "parking_lot_core" +version = "0.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09a279cbf25cb0757810394fbc1e359949b59e348145c643a939a525692e6929" +dependencies = [ + "cfg-if", + "libc", + "redox_syscall", + "smallvec", + "windows-sys", +] + +[[package]] +name = "pin-project-lite" +version = "0.2.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e0a7ae3ac2f1173085d398531c705756c94a4c56843785df85a60c1a0afac116" + +[[package]] +name = "pin-utils" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" + +[[package]] +name = "proc-macro-error" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da25490ff9892aab3fcf7c36f08cfb902dd3e71ca0f9f9517bea02a73a5ce38c" +dependencies = [ + "proc-macro-error-attr", + "proc-macro2", + "quote", + "syn", + "version_check", +] + +[[package]] +name = "proc-macro-error-attr" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1be40180e52ecc98ad80b184934baf3d0d29f979574e439af5a55274b35f869" +dependencies = [ + "proc-macro2", + "quote", + "version_check", +] + +[[package]] +name = "proc-macro2" +version = "1.0.43" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0a2ca2c61bc9f3d74d2886294ab7b9853abd9c1ad903a3ac7815c58989bb7bab" +dependencies = [ + "unicode-ident", +] + +[[package]] +name = "quote" +version = "1.0.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbe448f377a7d6961e30f5955f9b8d106c3f5e449d493ee1b125c1d43c2b5179" +dependencies = [ + "proc-macro2", +] + +[[package]] +name = "redox_syscall" +version = "0.2.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fb5a58c1855b4b6819d59012155603f0b22ad30cad752600aadfcb695265519a" +dependencies = [ + "bitflags", +] + +[[package]] +name = "redox_users" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b033d837a7cf162d7993aded9304e30a83213c648b6e389db233191f891e5c2b" +dependencies = [ + "getrandom", + "redox_syscall", + "thiserror", +] + +[[package]] +name = "regex" +version = "1.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4c4eb3267174b8c6c2f654116623910a0fef09c4753f8dd83db29c48a0df988b" +dependencies = [ + "aho-corasick", + "memchr", + "regex-syntax", +] + +[[package]] +name = "regex-syntax" +version = "0.6.27" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a3f87b73ce11b1619a3c6332f45341e0047173771e8b8b73f87bfeefb7b56244" + +[[package]] +name = "ryu" +version = "1.0.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4501abdff3ae82a1c1b477a17252eb69cee9e66eb915c1abaa4f44d873df9f09" + +[[package]] +name = "same-file" +version = "1.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93fc1dc3aaa9bfed95e02e6eadabb4baf7e3078b0bd1b4d7b6b0b68378900502" +dependencies = [ + "winapi-util", +] + +[[package]] +name = "scopeguard" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d29ab0c6d3fc0ee92fe66e2d99f700eab17a8d57d1c1d3b748380fb20baa78cd" + +[[package]] +name = "serde" +version = "1.0.144" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0f747710de3dcd43b88c9168773254e809d8ddbdf9653b84e2554ab219f17860" +dependencies = [ + "serde_derive", +] + +[[package]] +name = "serde_derive" +version = "1.0.144" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94ed3a816fb1d101812f83e789f888322c34e291f894f19590dc310963e87a00" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "serde_json" +version = "1.0.85" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e55a28e3aaef9d5ce0506d0a14dbba8054ddc7e499ef522dd8b26859ec9d4a44" +dependencies = [ + "itoa", + "ryu", + "serde", +] + +[[package]] +name = "signal-hook-registry" +version = "1.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e51e73328dc4ac0c7ccbda3a494dfa03df1de2f46018127f60c693f2648455b0" +dependencies = [ + "libc", +] + +[[package]] +name = "slab" +version = "0.4.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4614a76b2a8be0058caa9dbbaf66d988527d86d003c11a94fbd335d7661edcef" +dependencies = [ + "autocfg", +] + +[[package]] +name = "smallvec" +version = "1.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2fd0db749597d91ff862fd1d55ea87f7855a744a8425a64695b6fca237d1dad1" + +[[package]] +name = "socket2" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "10c98bba371b9b22a71a9414e420f92ddeb2369239af08200816169d5e2dd7aa" +dependencies = [ + "libc", + "winapi", +] + +[[package]] +name = "strsim" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "73473c0e59e6d5812c5dfe2a064a6444949f089e20eec9a2e5506596494e4623" + +[[package]] +name = "syn" +version = "1.0.99" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "58dbef6ec655055e20b86b15a8cc6d439cca19b667537ac6a1369572d151ab13" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + +[[package]] +name = "termcolor" +version = "1.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bab24d30b911b2376f3a13cc2cd443142f0c81dda04c118693e35b3835757755" +dependencies = [ + "winapi-util", +] + +[[package]] +name = "textwrap" +version = "0.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b1141d4d61095b28419e22cb0bbf02755f5e54e0526f97f1e3d1d160e60885fb" + +[[package]] +name = "thiserror" +version = "1.0.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f5f6586b7f764adc0231f4c79be7b920e766bb2f3e51b3661cdb263828f19994" +dependencies = [ + "thiserror-impl", +] + +[[package]] +name = "thiserror-impl" +version = "1.0.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "12bafc5b54507e0149cdf1b145a5d80ab80a90bcd9275df43d4fff68460f6c21" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "tokio" +version = "1.20.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a8325f63a7d4774dd041e363b2409ed1c5cbbd0f867795e661df066b2b0a581" +dependencies = [ + "autocfg", + "bytes", + "libc", + "memchr", + "mio", + "num_cpus", + "once_cell", + "parking_lot", + "pin-project-lite", + "signal-hook-registry", + "socket2", + "tokio-macros", + "winapi", +] + +[[package]] +name = "tokio-macros" +version = "1.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9724f9a975fb987ef7a3cd9be0350edcbe130698af5b8f7a631e23d42d052484" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "toml" +version = "0.5.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8d82e1a7758622a465f8cee077614c73484dac5b836c02ff6a40d5d1010324d7" +dependencies = [ + "serde", +] + +[[package]] +name = "unicode-ident" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c4f5b37a154999a8f3f98cc23a628d850e154479cd94decf3414696e12e31aaf" + +[[package]] +name = "uuid" +version = "1.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dd6469f4314d5f1ffec476e05f17cc9a78bc7a27a6a857842170bdf8d6f98d2f" +dependencies = [ + "getrandom", + "serde", +] + +[[package]] +name = "version_check" +version = "0.9.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "49874b5167b65d7193b8aba1567f5c7d93d001cafc34600cee003eda787e483f" + +[[package]] +name = "walkdir" +version = "2.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "808cf2735cd4b6866113f648b791c6adc5714537bc222d9347bb203386ffda56" +dependencies = [ + "same-file", + "winapi", + "winapi-util", +] + +[[package]] +name = "wasi" +version = "0.11.0+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" + +[[package]] +name = "winapi" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419" +dependencies = [ + "winapi-i686-pc-windows-gnu", + "winapi-x86_64-pc-windows-gnu", +] + +[[package]] +name = "winapi-i686-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" + +[[package]] +name = "winapi-util" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70ec6ce85bb158151cae5e5c87f95a8e97d2c0c4b001223f33a334e3ce5de178" +dependencies = [ + "winapi", +] + +[[package]] +name = "winapi-x86_64-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" + +[[package]] +name = "windows-sys" +version = "0.36.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ea04155a16a59f9eab786fe12a4a450e75cdb175f9e0d80da1e17db09f55b8d2" +dependencies = [ + "windows_aarch64_msvc", + "windows_i686_gnu", + "windows_i686_msvc", + "windows_x86_64_gnu", + "windows_x86_64_msvc", +] + +[[package]] +name = "windows_aarch64_msvc" +version = "0.36.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9bb8c3fd39ade2d67e9874ac4f3db21f0d710bee00fe7cab16949ec184eeaa47" + +[[package]] +name = "windows_i686_gnu" +version = "0.36.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "180e6ccf01daf4c426b846dfc66db1fc518f074baa793aa7d9b9aaeffad6a3b6" + +[[package]] +name = "windows_i686_msvc" +version = "0.36.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e2e7917148b2812d1eeafaeb22a97e4813dfa60a3f8f78ebe204bcc88f12f024" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.36.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4dcd171b8776c41b97521e5da127a2d86ad280114807d0b2ab1e462bc764d9e1" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.36.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c811ca4a8c853ef420abd8592ba53ddbbac90410fab6903b3e79972a631f7680" diff --git a/Cargo.toml b/Cargo.toml new file mode 100644 index 0000000..35b3d1b --- /dev/null +++ b/Cargo.toml @@ -0,0 +1,19 @@ +[package] +name = "media-linter" +version = "0.1.0" +edition = "2021" + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[dependencies] +eyre = "0.6" +tokio = { version = "1", features = ["full"] } +futures = "0.3" +serde = { version = "1.0", features = ["derive"] } +serde_json = "1.0" +clap = { version = "3.2", features = ["derive"] } +toml = "0.5" +dirs = "4.0" +uuid = { version = "1.1", features = ["serde", "v4"] } +regex = "1.6" +walkdir = "2" diff --git a/readme.md b/readme.md new file mode 100644 index 0000000..29e447b --- /dev/null +++ b/readme.md @@ -0,0 +1,14 @@ +# Media Linter + +A tool to lint the file names of your media + +## Filenames + +| Type | Filename | +|-------|--------------------------------------| +| Show | Show Name S00E00 - Episode Name.ext | +| Movie | Movie Name \(year) \[resolution].ext | + +## Rules + +... diff --git a/src/args.rs b/src/args.rs new file mode 100644 index 0000000..b4ebd34 --- /dev/null +++ b/src/args.rs @@ -0,0 +1,14 @@ +use clap::Parser; + +#[derive(Parser, Debug)] +#[clap(author, version, about, long_about = None)] +pub struct Args { + #[clap(long, value_parser)] + pub no_emoji: bool, + + #[clap(long, value_parser)] + pub show_success: bool, + + #[clap(long, value_parser)] + pub path: String, +} \ No newline at end of file diff --git a/src/main.rs b/src/main.rs new file mode 100644 index 0000000..9e8468c --- /dev/null +++ b/src/main.rs @@ -0,0 +1,120 @@ +mod args; +mod rules; +mod utils; + +use std::fmt::{Display, Formatter}; +use eyre::Result; +use clap::Parser; +use walkdir::{DirEntry, WalkDir}; + +use args::Args; +use rules::*; +use utils::status_icon::*; + +const VIDEO_EXTENSIONS: [&str; 14] = ["mkv", "mp4", "avi", "webm", "mov", "wmv", "flv", "ogg", "ogv", "yuv", "amv", "mpg", "mpeg", "m4v"]; +const SERIES_REGEX: &str = r"^(?P.+)\sS(?P<season>\d\d)E(?P<episode>\d\d\d?)(E(?P<episode2>\d\d\d?))?((?P<nameSeparator>\s-\s)?(?P<name>.+))?\.(?P<ext>...)$"; +// const FILE_EXT_REGEX: &str = r"^(?P<title>.+)\.(?P<ext>...)$"; +// const MOVIE_REGEX: &str = r"^(?P<title>.+)\s(?P<year>\(\d{4}\))\s(?P<resolution>\[.+\])\.(?P<ext>...)$"; + +struct Stats { + success: usize, + warning: usize, + error: usize, + files: usize, +} + +impl Display for Stats { + fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { + write!( + f, + "Files: {} - Success: {} - Warnings: {} - Errors: {}", + self.files, + self.success, + self.warning, + self.error) + } +} + +#[tokio::main] +async fn main() -> Result<()> { + let params = Args::parse(); + + println!("{}", params.path); + + let mut stats = Stats { + files: 0, + success: 0, + warning: 0, + error: 0, + }; + + for entry in WalkDir::new(params.path) { + if let Ok(file) = entry { + check_file(file, &mut stats, params.no_emoji, params.show_success); + } + } + + println!("{}", stats); + + Ok(()) +} + +fn check_file(file: DirEntry, stats: &mut Stats, no_emoji: bool, show_success: bool) -> () { + let file_type = file.file_type(); + if file_type.is_file() { + stats.files += 1; + let filename = file.file_name().to_str().unwrap(); + match lint_file_name(&file, filename) { + ComplianceStatus::NotMatched => { + stats.error += 1; + println!("{} {} --- (Failed to match)", get_status_icon(StatusIcon::Failure, no_emoji), filename) + } + ComplianceStatus::NonCompliant(reason) => { + stats.warning += 1; + println!("{} {} --- ({})", get_status_icon(StatusIcon::Warning, no_emoji), filename, reason); + } + ComplianceStatus::Compliant => { + stats.success += 1; + if show_success { + println!("{} {}", get_status_icon(StatusIcon::Success, no_emoji), filename) + } + } + } + } else if file_type.is_dir() { + println!("\n\n=== {} {}", get_status_icon(StatusIcon::Directory, no_emoji), file.path().display()) + } +} + +fn lint_file_name(file: &DirEntry, filename: &str) -> ComplianceStatus { + if let Some(ext) = file.path().extension() { + if !VIDEO_EXTENSIONS.contains(&ext.to_str().unwrap()) { + return ComplianceStatus::Compliant; + } + } + + let captures = regex::Regex::new(SERIES_REGEX).unwrap().captures(filename); + + if captures.is_none() { + return ComplianceStatus::NotMatched; + } + + let captures = captures.unwrap(); + + if let Some(reason) = HasEpisodeName::check(filename, &captures) { + return ComplianceStatus::NonCompliant(reason); + } + + if let Some(reason) = MissingSeparator::check(filename, &captures) { + return ComplianceStatus::NonCompliant(reason); + } + + if let Some(reason) = DashInTitle::check(filename, &captures) { + return ComplianceStatus::NonCompliant(reason); + } + + if let Some(reason) = HasFluff::check(filename, &captures) { + return ComplianceStatus::NonCompliant(reason); + } + + ComplianceStatus::Compliant +} \ No newline at end of file diff --git a/src/rules/has_dash_in_title.rs b/src/rules/has_dash_in_title.rs new file mode 100644 index 0000000..7a59f7b --- /dev/null +++ b/src/rules/has_dash_in_title.rs @@ -0,0 +1,20 @@ +use regex::Captures; +use eyre::Result; +use super::*; + +pub struct DashInTitle {} +impl Rule for DashInTitle { + fn check(_filename: &str, captures: &Captures) -> Option<NonCompliantReason> { + let title = captures.name("title").unwrap(); + + if title.as_str().contains("-") { + return Some(NonCompliantReason::DashInTitle) + } + + return None + } + + fn fix(_filename: &str, _captures: &Captures) -> Result<FixStatus> { + return Ok(FixStatus::NotImplemented) + } +} \ No newline at end of file diff --git a/src/rules/has_episode_name.rs b/src/rules/has_episode_name.rs new file mode 100644 index 0000000..6558c31 --- /dev/null +++ b/src/rules/has_episode_name.rs @@ -0,0 +1,18 @@ +use regex::Captures; +use eyre::Result; +use super::*; + +pub struct HasEpisodeName {} +impl Rule for HasEpisodeName { + fn check(_filename: &str, captures: &Captures) -> Option<NonCompliantReason> { + if captures.name("name").is_none() { + return Some(NonCompliantReason::MissingName) + } + + return None + } + + fn fix(_filename: &str, _captures: &Captures) -> Result<FixStatus> { + return Ok(FixStatus::NotImplemented) + } +} diff --git a/src/rules/has_fluff.rs b/src/rules/has_fluff.rs new file mode 100644 index 0000000..77465de --- /dev/null +++ b/src/rules/has_fluff.rs @@ -0,0 +1,24 @@ +use regex::Captures; +use eyre::Result; +use super::*; + +const KNOWN_FLUFF: [&str; 7] = ["[", "xvid", "BluRay", "1080p", "264", "dvdrip", "web-dl"]; + +pub struct HasFluff {} +impl Rule for HasFluff { + fn check(filename: &str, _captures: &Captures) -> Option<NonCompliantReason> { + let lowered_filename = filename.to_lowercase(); + for fluff in KNOWN_FLUFF { + if lowered_filename.contains(fluff) { + return Some(NonCompliantReason::HasFluff) + } + } + + return None + } + + fn fix(_filename: &str, _captures: &Captures) -> Result<FixStatus> { + return Ok(FixStatus::NotImplemented) + } +} + diff --git a/src/rules/missing_separator.rs b/src/rules/missing_separator.rs new file mode 100644 index 0000000..4b7fea9 --- /dev/null +++ b/src/rules/missing_separator.rs @@ -0,0 +1,18 @@ +use regex::Captures; +use eyre::Result; +use super::*; + +pub struct MissingSeparator {} +impl Rule for MissingSeparator { + fn check(_filename: &str, captures: &Captures) -> Option<NonCompliantReason> { + if captures.name("nameSeparator").is_none() { + return Some(NonCompliantReason::MissingNameSeparator) + } + + return None + } + + fn fix(_filename: &str, _captures: &Captures) -> Result<FixStatus> { + return Ok(FixStatus::NotImplemented) + } +} diff --git a/src/rules/mod.rs b/src/rules/mod.rs new file mode 100644 index 0000000..f59e2bb --- /dev/null +++ b/src/rules/mod.rs @@ -0,0 +1,48 @@ +use std::fmt::{Display, Formatter}; +use regex::Captures; +use eyre::Result; + +mod has_fluff; +mod has_dash_in_title; +mod has_episode_name; +mod missing_separator; + +pub use has_fluff::HasFluff; +pub use has_episode_name::HasEpisodeName; +pub use has_dash_in_title::DashInTitle; +pub use missing_separator::MissingSeparator; + +pub enum NonCompliantReason { + DashInTitle, + MissingName, + HasFluff, + MissingNameSeparator +} + +impl Display for NonCompliantReason { + fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { + match self { + NonCompliantReason::DashInTitle => write!(f, "Title contains dash"), + NonCompliantReason::MissingName => write!(f, "Missing episode name"), + NonCompliantReason::HasFluff => write!(f, "Has fluff"), + NonCompliantReason::MissingNameSeparator => write!(f, "Missing episode name separator"), + } + } +} + +pub enum ComplianceStatus { + NonCompliant(NonCompliantReason), + NotMatched, + Compliant, +} + +pub enum FixStatus { + // Fixed, + NotImplemented, + // NotFixable +} + +pub trait Rule { + fn check(filename: &str, captures: &Captures) -> Option<NonCompliantReason>; + fn fix(filename: &str, captures: &Captures) -> Result<FixStatus>; +} diff --git a/src/utils/constructors.rs b/src/utils/constructors.rs new file mode 100644 index 0000000..5cf92c1 --- /dev/null +++ b/src/utils/constructors.rs @@ -0,0 +1,4 @@ +#[allow(dead_code)] +pub fn episode_name(title: String, season: String, episode: String, name: String, format: String) -> String { + format!("${title} S${season}E${episode} - ${name}.${format}") +} diff --git a/src/utils/mod.rs b/src/utils/mod.rs new file mode 100644 index 0000000..7ebbf09 --- /dev/null +++ b/src/utils/mod.rs @@ -0,0 +1,2 @@ +pub mod constructors; +pub mod status_icon; diff --git a/src/utils/status_icon.rs b/src/utils/status_icon.rs new file mode 100644 index 0000000..10ba0b9 --- /dev/null +++ b/src/utils/status_icon.rs @@ -0,0 +1,15 @@ +pub enum StatusIcon { + Failure, + Warning, + Success, + Directory, +} + +pub fn get_status_icon(icon: StatusIcon, no_emoji: bool) -> String { + match icon { + StatusIcon::Failure => { if no_emoji { "F -".to_string() } else { "❌".to_string() } } + StatusIcon::Warning => { if no_emoji { "W -".to_string() } else { "⚠️".to_string() } } + StatusIcon::Success => { if no_emoji { "S -".to_string() } else { "✔️".to_string() } } + StatusIcon::Directory => { if no_emoji { "".to_string() } else { "📂".to_string() } } + } +} \ No newline at end of file