From 1a26c40191bda843a0500f12bbb7d67b3e8c238e Mon Sep 17 00:00:00 2001 From: Baitinq Date: Sun, 30 Oct 2022 23:08:16 +0100 Subject: Indexer: Use kuchiki to split html content into words This is better than html2text when using non-ascii characters. --- indexer/Cargo.toml | 1 + indexer/src/main.rs | 24 ++++++++++++++++++------ 2 files changed, 19 insertions(+), 6 deletions(-) (limited to 'indexer') diff --git a/indexer/Cargo.toml b/indexer/Cargo.toml index 28e6f17..2c8f905 100644 --- a/indexer/Cargo.toml +++ b/indexer/Cargo.toml @@ -11,6 +11,7 @@ actix-cors = "0.6.3" scraper = "0.12.0" html2text = "0.4.3" serde_json = "1.0.87" +kuchiki = "0.8.1" lib = { path = "../lib" } [[bin]] diff --git a/indexer/src/main.rs b/indexer/src/main.rs index 36bd8de..1df3cf5 100644 --- a/indexer/src/main.rs +++ b/indexer/src/main.rs @@ -3,6 +3,7 @@ use actix_web::{get, post, web, App, HttpServer, Responder}; use std::collections::{HashMap, HashSet}; use std::sync::{Arc, Mutex}; use lib::lib::*; +use kuchiki::traits::TendrilSink; struct AppState { database: Mutex>>, @@ -41,17 +42,28 @@ async fn add_resource( ) -> impl Responder { //parse content let document = scraper::Html::parse_document(resource.content.as_str()); - - //TODO: Not very good, can we just body.get_text()? - let text = html2text::from_read(resource.content.as_str().as_bytes(), resource.content.len()); + let kuchiki_parser = kuchiki::parse_html().one(resource.content.as_str()); + + //remove script, style and noscript tags + kuchiki_parser + .inclusive_descendants() + .filter(|node| { + node.as_element().map_or(false, |e| { + matches!(e.name.local.as_ref(), "script" | "style" | "noscript") + }) + }) + .collect::>() + .iter() + .for_each(|node| node.detach()); + + let text = kuchiki_parser.text_contents(); let split_words = text.split(' '); //fixup words (remove words with non alphabetic chars, empty words, transform to lowercase...) let fixed_words: Vec = split_words - .filter(|w| !w.chars().any(|c| !c.is_ascii_alphabetic())) - .filter(|w| !w.is_empty() && *w != " ") - .map(|w| w.to_ascii_lowercase()) + .map(|w| w.to_ascii_lowercase().split_whitespace().collect()) + .filter(|w: &String| !w.is_empty()) .collect(); println!("xd: {:?}", fixed_words); -- cgit 1.4.1