aboutsummaryrefslogtreecommitdiff
path: root/src/url.rs
blob: 5289d285884403a338b659e63991d643d214d2ef (plain) (blame)
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
use url::Url;

pub fn from_path(
  path: &str,
  fallback: bool,
  configuration: &mut crate::response::configuration::Configuration,
) -> Result<Url, url::ParseError> {
  Url::try_from(&*if path.starts_with("/proxy") {
    configuration.set_proxy(true);

    format!(
      "gemini://{}{}",
      path.replace("/proxy/", ""),
      if fallback { "/" } else { "" }
    )
  } else if path.starts_with("/x") {
    configuration.set_proxy(true);

    format!(
      "gemini://{}{}",
      path.replace("/x/", ""),
      if fallback { "/" } else { "" }
    )
  } else if path.starts_with("/raw") {
    configuration.set_proxy(true);
    configuration.set_raw(true);

    format!(
      "gemini://{}{}",
      path.replace("/raw/", ""),
      if fallback { "/" } else { "" }
    )
  } else if path.starts_with("/nocss") {
    configuration.set_proxy(true);
    configuration.set_no_css(true);

    format!(
      "gemini://{}{}",
      path.replace("/nocss/", ""),
      if fallback { "/" } else { "" }
    )
  } else {
    format!(
      "{}{}{}",
      {
        std::env::var("ROOT").unwrap_or_else(|_| {
          warn!(
            "could not use ROOT from environment variables, proceeding with \
             default root: gemini://fuwn.me"
          );

          "gemini://fuwn.me".to_string()
        })
      },
      path,
      if fallback { "/" } else { "" }
    )
  })
}

pub fn matches_pattern(pattern: &str, path: &str) -> bool {
  if !pattern.contains('*') {
    return path == pattern;
  }

  let parts: Vec<&str> = pattern.split('*').collect();
  let mut position = if pattern.starts_with('*') {
    0
  } else {
    let first = parts.first().unwrap();

    if !path.starts_with(first) {
      return false;
    }

    first.len()
  };
  let before_last = parts.len().saturating_sub(1);

  for part in &parts[1..before_last] {
    if part.is_empty() {
      continue;
    }

    if let Some(found) = path[position..].find(part) {
      position += found + part.len();
    } else {
      return false;
    }
  }

  if !pattern.ends_with('*') {
    let last = parts.last().unwrap();

    if !path[position..].ends_with(last) {
      return false;
    }
  }

  true
}