mirror of
https://github.com/xvxx/phd
synced 2024-11-10 13:10:23 +00:00
serve gophermaps
This commit is contained in:
parent
d039060a80
commit
981e304721
@ -8,6 +8,8 @@
|
|||||||
|
|
||||||
an esoteric gopher server.
|
an esoteric gopher server.
|
||||||
|
|
||||||
|
point it at a directory and it'll serve up all its text files, sub-directories, and binary files over gopher.
|
||||||
|
|
||||||
## todo
|
## todo
|
||||||
|
|
||||||
- [ ] index.gph
|
- [ ] index.gph
|
||||||
|
@ -1,10 +1,11 @@
|
|||||||
use crate::{Request, Result};
|
use crate::{Request, Result};
|
||||||
use gophermap::{GopherMenu, ItemType};
|
use gophermap::{GopherMenu, ItemType};
|
||||||
use std::{
|
use std::{
|
||||||
fs,
|
fs::{self, File},
|
||||||
io::prelude::*,
|
io::prelude::*,
|
||||||
io::{BufReader, Read, Write},
|
io::{BufReader, Read, Write},
|
||||||
net::{TcpListener, TcpStream},
|
net::{TcpListener, TcpStream},
|
||||||
|
path::Path,
|
||||||
};
|
};
|
||||||
use threadpool::ThreadPool;
|
use threadpool::ThreadPool;
|
||||||
|
|
||||||
@ -46,12 +47,39 @@ fn accept(stream: TcpStream, mut req: Request) -> Result<()> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Writes a response to a client based on a Request.
|
/// Writes a response to a client based on a Request.
|
||||||
fn write_response<'a, W>(w: &'a W, req: Request) -> Result<()>
|
fn write_response<'a, W>(w: &'a W, mut req: Request) -> Result<()>
|
||||||
where
|
where
|
||||||
&'a W: Write,
|
&'a W: Write,
|
||||||
{
|
{
|
||||||
let md = fs::metadata(&req.file_path())?;
|
let path = req.file_path();
|
||||||
if md.is_file() {
|
|
||||||
|
// check for dir.gph if we're looking for dir
|
||||||
|
let mut gph_file = path.clone();
|
||||||
|
gph_file.push_str(".gph");
|
||||||
|
if Path::new(&gph_file).exists() {
|
||||||
|
req.selector = req.selector.trim_end_matches('/').into();
|
||||||
|
req.selector.push_str(".gph");
|
||||||
|
return write_gophermap(w, req);
|
||||||
|
} else {
|
||||||
|
// check for index.gph if we're looking for dir
|
||||||
|
let mut index = path.clone();
|
||||||
|
if !index.ends_with('/') {
|
||||||
|
index.push('/');
|
||||||
|
}
|
||||||
|
index.push_str("index.gph");
|
||||||
|
if Path::new(&index).exists() {
|
||||||
|
if !req.selector.ends_with('/') {
|
||||||
|
req.selector.push('/');
|
||||||
|
}
|
||||||
|
req.selector.push_str("index.gph");
|
||||||
|
return write_gophermap(w, req);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let md = fs::metadata(&path)?;
|
||||||
|
if path.ends_with(".gph") {
|
||||||
|
write_gophermap(w, req)
|
||||||
|
} else if md.is_file() {
|
||||||
write_file(w, req)
|
write_file(w, req)
|
||||||
} else if md.is_dir() {
|
} else if md.is_dir() {
|
||||||
write_dir(w, req)
|
write_dir(w, req)
|
||||||
@ -105,6 +133,27 @@ where
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Send a gophermap (menu) to the client based on a Request.
|
||||||
|
fn write_gophermap<'a, W>(mut w: &'a W, req: Request) -> Result<()>
|
||||||
|
where
|
||||||
|
&'a W: Write,
|
||||||
|
{
|
||||||
|
let path = req.file_path();
|
||||||
|
let file = File::open(&path)?;
|
||||||
|
let reader = BufReader::new(file);
|
||||||
|
for line in reader.lines() {
|
||||||
|
let mut line = line?.trim_end_matches("\r\n").to_string();
|
||||||
|
match line.chars().filter(|&c| c == '\t').count() {
|
||||||
|
1 => line.push_str(&format!("\t{}\t{}", req.host, req.port)),
|
||||||
|
2 => line.push_str(&format!("\t{}", req.port)),
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
line.push_str("\r\n");
|
||||||
|
w.write_all(line.as_bytes())?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
/// Determine the gopher type for a DirEntry on disk.
|
/// Determine the gopher type for a DirEntry on disk.
|
||||||
fn file_type(dir: &fs::DirEntry) -> ItemType {
|
fn file_type(dir: &fs::DirEntry) -> ItemType {
|
||||||
let metadata = match dir.metadata() {
|
let metadata = match dir.metadata() {
|
||||||
|
Loading…
Reference in New Issue
Block a user