~/website

website/src/main.rs -rw-r--r-- 6 kB
5b96ebd5 — arthurmelton working 2 years ago
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
use walkdir::WalkDir;
use std::fs::{self, File};
use std::path::{Path, PathBuf};
use handlebars::Handlebars;
use std::io::Write;
use toml::{Table, Value};
use chrono::NaiveDate;
use serde_derive::Serialize;

#[derive(Clone, Serialize)]
struct Blog {
    config: Table,
    path: PathBuf
}

#[derive(Clone, Serialize)]
struct BlogPage {
    title: String,
    blogs: Vec<Blog>,
    before: Option<usize>,
    after: Option<usize>
}

fn main() {
    // make public folder
    let _ = fs::remove_dir_all("public");
    fs::create_dir("public").expect("could not make the folder \"public\"");

    //copy all static
    for e in WalkDir::new("static").into_iter().filter_map(|e| e.ok()) {
        if e.metadata().unwrap().is_file() {
            let mut path = e.path();
            path = path.strip_prefix("static/").unwrap(); // should never fail
            fs::copy(e.path(), Path::new("public").join(path)).expect(&format!("failed to copy static/{} to public/{}", path.display(), path.display()));
        }
    }

    // make template
    let mut reg = Handlebars::new();
    reg.register_template_string("main", fs::read_to_string("template.html.hbs").expect("cant read template.html.hbs")).expect("cant make template");

    //format all pages
    for e in WalkDir::new("pages").into_iter().filter_map(|e| e.ok()) {
        if e.metadata().unwrap().is_file() {
            let path = e.path();
            let contents = fs::read_to_string(path).expect(&format!("Cant read file {}", path.display()));
            let mut config = format!("{}\ncontent = \"\"", contents.split("+++").nth(1).expect(&format!("{} does not have a +++ section", path.display()))).parse::<Table>().unwrap();
            let content = contents.split("+++").nth(2).expect(&format!("{} does not have a +++ section", path.display()));
            let config_content = config.get_mut("content").unwrap();
            *config_content = Value::try_from(md_to_html(content)).unwrap();
            let path = Path::new("public").join(path.strip_prefix("pages").unwrap()).with_extension("html");
            let mut file = File::create(path.clone()).expect(&format!("Cant make file {}", path.display()));
            file.write_all(reg.render("main", &config).expect(&format!("Cant render {}", path.display())).as_bytes()).expect(&format!("Cant write to file {}", path.display()))
        }
    }

    //do the blogs
    match fs::read_dir("blogs") {
        Ok(blogs) => {
            fs::create_dir("public/blogs").expect("could not make the folder \"public/blogs\"");
            let mut all_blogs:Vec<Blog> = Vec::new();
            for i in blogs {
                if let Ok(i) = i {
                let path = i.path();
                    let contents = fs::read_to_string(path.clone()).expect(&format!("Cant read file {}", path.display()));
                    let mut config = format!("{}\ncontent = \"\"", contents.split("+++").nth(1).expect(&format!("{} does not have a +++ section", path.display()))).parse::<Table>().unwrap();
                    let content = contents.split("+++").nth(2).expect(&format!("{} does not have a +++ section", path.display()));
                    let config_content = config.get_mut("content").unwrap();
                    *config_content = Value::try_from(md_to_html(content)).unwrap();
                    let path = path.strip_prefix("blogs").unwrap().with_extension("html");
                    let mut file = File::create(Path::new("public/blogs").join(path.clone())).expect(&format!("Cant make file public/blogs/{}", path.display()));
                    file.write_all(reg.render("main", &config).expect(&format!("Cant render public/blogs/{}", path.display())).as_bytes()).expect(&format!("Cant write to file public/blogs/{}", path.display()));
                    all_blogs.push(Blog {
                        config: config,
                        path: path.to_path_buf()
                    });
                }
            }
            all_blogs.sort_by(|a,b| NaiveDate::parse_from_str(b.config["date"].as_str().expect(&format!("{} does not have a date", b.path.display())), "%Y-%m-%d").expect(&format!("Cant convert {} date", b.path.display())).cmp(&NaiveDate::parse_from_str(a.config["date"].as_str().expect(&format!("{} does not have a date", a.path.display())), "%Y-%m-%d").expect(&format!("Cant convert {} date", a.path.display()))));
            let mut i = 0;
            let blog_pages = all_blogs.chunks(10).map(|x| {
                let before = if i > 0 {
                    Some(i-1)
                }
                else {
                    None
                };
                let after = if i < all_blogs.len()/10 {
                    Some(i+1)
                }
                else {
                    None
                };
                i+=1;
                BlogPage {
                    title: "Blogs".to_string(),
                    blogs: x.to_vec(),
                    before,
                    after
                }
            }).collect::<Vec<BlogPage>>();
            let mut x = 0;
            for i in blog_pages {
                let path_name = format!("public/blogs-{}.html", x);
                let path = Path::new(&path_name);
                let mut file = File::create(path.clone()).expect(&format!("Cant make file {}", path.display()));
                file.write_all(reg.render("main", &i).expect(&format!("Cant render {}", path.display())).as_bytes()).expect(&format!("Cant write to file {}", path.display()));
                x+=1;
            }

        },
        Err(_) => {}
    }
}

use emojicons::EmojiFormatter;
use pulldown_cmark::{html, Options, Parser};

pub fn md_to_html(input: &str) -> String {
    let input = EmojiFormatter(input).to_string();
    let mut options = Options::empty();
    options.insert(Options::ENABLE_STRIKETHROUGH);
    let parser = Parser::new_ext(&input, options);
    let mut html_output = String::new();
    html::push_html(&mut html_output, parser);
    html_output
}