Start of reading album from a directory, including finding image files
This commit is contained in:
parent
0936ce2069
commit
4272a22a34
7 changed files with 1136 additions and 123 deletions
922
Cargo.lock
generated
922
Cargo.lock
generated
File diff suppressed because it is too large
Load diff
|
@ -10,6 +10,7 @@ edition = "2024"
|
||||||
[dependencies]
|
[dependencies]
|
||||||
anyhow = "^1.0"
|
anyhow = "^1.0"
|
||||||
clap = { version = "^4.5", features = ["derive"] }
|
clap = { version = "^4.5", features = ["derive"] }
|
||||||
|
image = "^0.25.6"
|
||||||
serde = { version = "^1.0", features = ["derive"] }
|
serde = { version = "^1.0", features = ["derive"] }
|
||||||
serde_yml = "^0.0.12"
|
serde_yml = "^0.0.12"
|
||||||
thiserror = "^2.0"
|
thiserror = "^2.0"
|
||||||
|
|
122
src/generate.rs
122
src/generate.rs
|
@ -1,118 +1,10 @@
|
||||||
use std::path::PathBuf;
|
mod album_dir;
|
||||||
|
|
||||||
/// An album directory, which has images and possibly child albums
|
use album_dir::AlbumDir;
|
||||||
#[derive(Clone)]
|
use std::io;
|
||||||
struct AlbumDir {
|
use std::path::{Path, PathBuf};
|
||||||
path: PathBuf,
|
|
||||||
images: Vec<Image>,
|
|
||||||
// TOOD: Remove the parent reference? Causes a lot of issues
|
|
||||||
// parent: Option<Box<&'a AlbumDir>>,
|
|
||||||
children: Vec<AlbumDir>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl AlbumDir {
|
pub fn generate(root_path: &PathBuf) -> Result<(), io::Error> {
|
||||||
fn iter(&self) -> AlbumIter {
|
let _ = AlbumDir::try_from(root_path)?;
|
||||||
AlbumIter::new(self)
|
Ok(())
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: from-path
|
|
||||||
|
|
||||||
/// An iterator which walks through all of the images in an album, and its sub-albums
|
|
||||||
struct AlbumIter<'a> {
|
|
||||||
root_album: &'a AlbumDir,
|
|
||||||
image_iter: Box<dyn Iterator<Item = &'a Image> + 'a>,
|
|
||||||
children_iter: std::slice::Iter<'a, AlbumDir>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a> AlbumIter<'a> {
|
|
||||||
fn new(ad: &'a AlbumDir) -> Self {
|
|
||||||
Self {
|
|
||||||
root_album: ad,
|
|
||||||
image_iter: Box::new(ad.images.iter()),
|
|
||||||
children_iter: ad.children.iter(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a> Iterator for AlbumIter<'a> {
|
|
||||||
type Item = &'a Image;
|
|
||||||
|
|
||||||
fn next(&mut self) -> Option<Self::Item> {
|
|
||||||
if let Some(img) = self.image_iter.next() {
|
|
||||||
return Some(img);
|
|
||||||
}
|
|
||||||
|
|
||||||
for album in self.children_iter.by_ref() {
|
|
||||||
// Set the child album as the current image iterator
|
|
||||||
self.image_iter = Box::new(album.iter());
|
|
||||||
// If we found a child album with an image, return the image. Otherwise we'll keep
|
|
||||||
// iterating over children.
|
|
||||||
if let Some(i) = self.image_iter.next() {
|
|
||||||
return Some(i);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Hash, PartialEq, Eq)]
|
|
||||||
struct Image {
|
|
||||||
path: PathBuf,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(test)]
|
|
||||||
mod tests {
|
|
||||||
use super::*;
|
|
||||||
use std::collections::HashSet;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn basic_album_iter() {
|
|
||||||
let mut ad = AlbumDir {
|
|
||||||
path: "".into(),
|
|
||||||
images: vec![Image { path: "foo".into() }, Image { path: "bar".into() }],
|
|
||||||
children: vec![],
|
|
||||||
};
|
|
||||||
// A child album with some images
|
|
||||||
ad.children.push(AlbumDir {
|
|
||||||
path: "subdir".into(),
|
|
||||||
images: vec![
|
|
||||||
Image {
|
|
||||||
path: "subdir/foo".into(),
|
|
||||||
},
|
|
||||||
Image {
|
|
||||||
path: "subdir/bar".into(),
|
|
||||||
},
|
|
||||||
],
|
|
||||||
children: vec![AlbumDir {
|
|
||||||
path: "deeper_subdir".into(),
|
|
||||||
images: vec![Image {
|
|
||||||
path: "deeper_subdir/image.jpg".into(),
|
|
||||||
}],
|
|
||||||
children: vec![],
|
|
||||||
}],
|
|
||||||
});
|
|
||||||
// A child album with no images
|
|
||||||
ad.children.push(AlbumDir {
|
|
||||||
path: "another_subdir".into(),
|
|
||||||
images: vec![],
|
|
||||||
children: vec![],
|
|
||||||
});
|
|
||||||
|
|
||||||
let imgs: HashSet<String> = ad
|
|
||||||
.iter()
|
|
||||||
.map(|i| i.path.clone().to_str().unwrap().to_string())
|
|
||||||
.collect();
|
|
||||||
let expected: HashSet<String> = [
|
|
||||||
"foo",
|
|
||||||
"bar",
|
|
||||||
"subdir/foo",
|
|
||||||
"subdir/bar",
|
|
||||||
"deeper_subdir/image.jpg",
|
|
||||||
]
|
|
||||||
.iter()
|
|
||||||
.map(|s| s.to_string())
|
|
||||||
.collect();
|
|
||||||
assert_eq!(imgs, expected);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
205
src/generate/album_dir.rs
Normal file
205
src/generate/album_dir.rs
Normal file
|
@ -0,0 +1,205 @@
|
||||||
|
use image::ImageReader;
|
||||||
|
use std::ffi::OsString;
|
||||||
|
use std::io;
|
||||||
|
use std::path::PathBuf;
|
||||||
|
use std::slice::Iter;
|
||||||
|
|
||||||
|
/// An album directory, which has images and possibly child albums
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct AlbumDir {
|
||||||
|
path: PathBuf,
|
||||||
|
images: Vec<Image>,
|
||||||
|
// TOOD: Remove the parent reference? Causes a lot of issues
|
||||||
|
// parent: Option<Box<&'a AlbumDir>>,
|
||||||
|
children: Vec<AlbumDir>,
|
||||||
|
|
||||||
|
description: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AlbumDir {
|
||||||
|
/// Returns an iterator over all images in the album and subalbums
|
||||||
|
fn iter(&self) -> AlbumIter {
|
||||||
|
AlbumIter::new(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TryFrom<&PathBuf> for AlbumDir {
|
||||||
|
type Error = io::Error;
|
||||||
|
|
||||||
|
fn try_from(p: &PathBuf) -> io::Result<AlbumDir> {
|
||||||
|
let mut images = vec![];
|
||||||
|
let mut children = vec![];
|
||||||
|
let mut description = "".to_string();
|
||||||
|
|
||||||
|
for entry in p.read_dir()? {
|
||||||
|
let entry_path = entry?.path();
|
||||||
|
|
||||||
|
if entry_path.is_file() {
|
||||||
|
println!("Found file: {}", entry_path.display());
|
||||||
|
if let Some(filename) = entry_path.file_name() {
|
||||||
|
if filename == "description.txt" {
|
||||||
|
todo!();
|
||||||
|
// description = String::from_utf8(fs::read(entry_path)?)?;
|
||||||
|
} else if filename == "description.md" {
|
||||||
|
todo!();
|
||||||
|
} else {
|
||||||
|
let reader = ImageReader::open(&entry_path)?.with_guessed_format()?;
|
||||||
|
if reader.format().is_some() {
|
||||||
|
// Found an image
|
||||||
|
// TODO: If image filename but with .md or .txt exists, read that in as
|
||||||
|
// the image description OR make this part of Image::from<Path>
|
||||||
|
todo!();
|
||||||
|
let description = String::new();
|
||||||
|
images.push(Image {
|
||||||
|
filename: filename.to_os_string(),
|
||||||
|
description,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if entry_path.is_dir() {
|
||||||
|
println!("Found dir: {}", entry_path.display());
|
||||||
|
|
||||||
|
if let Some(dirname) = entry_path.file_name().and_then(|n| n.to_str()) {
|
||||||
|
if dirname.starts_with("_") {
|
||||||
|
// Likely a templates or static dir
|
||||||
|
continue;
|
||||||
|
} else if dirname == "site" {
|
||||||
|
// Is a generated site dir, don't descend into it
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
children.push(AlbumDir::try_from(&entry_path)?);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find all directories in directory and make AlbumDirs out of them,
|
||||||
|
// but skip dirs known to have interesting stuff
|
||||||
|
Ok(AlbumDir {
|
||||||
|
path: p.clone(),
|
||||||
|
images,
|
||||||
|
children,
|
||||||
|
description,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: from-path, find all images and children
|
||||||
|
|
||||||
|
/// An iterator which walks through all of the images in an album, and its sub-albums
|
||||||
|
struct AlbumIter<'a> {
|
||||||
|
root_album: &'a AlbumDir,
|
||||||
|
image_iter: Box<dyn Iterator<Item = &'a Image> + 'a>,
|
||||||
|
children_iter: Iter<'a, AlbumDir>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AlbumIter<'a> {
|
||||||
|
fn new(ad: &'a AlbumDir) -> Self {
|
||||||
|
Self {
|
||||||
|
root_album: ad,
|
||||||
|
image_iter: Box::new(ad.images.iter()),
|
||||||
|
children_iter: ad.children.iter(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> Iterator for AlbumIter<'a> {
|
||||||
|
type Item = &'a Image;
|
||||||
|
|
||||||
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
|
if let Some(img) = self.image_iter.next() {
|
||||||
|
return Some(img);
|
||||||
|
}
|
||||||
|
|
||||||
|
for album in self.children_iter.by_ref() {
|
||||||
|
// Set the child album as the current image iterator
|
||||||
|
self.image_iter = Box::new(album.iter());
|
||||||
|
// If we found a child album with an image, return the image. Otherwise we'll keep
|
||||||
|
// iterating over children.
|
||||||
|
if let Some(i) = self.image_iter.next() {
|
||||||
|
return Some(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Hash, PartialEq, Eq)]
|
||||||
|
struct Image {
|
||||||
|
filename: OsString,
|
||||||
|
description: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
use std::collections::HashSet;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn basic_album_iter() {
|
||||||
|
let mut ad = AlbumDir {
|
||||||
|
path: "".into(),
|
||||||
|
description: "".to_string(),
|
||||||
|
images: vec![
|
||||||
|
Image {
|
||||||
|
path: "foo".into(),
|
||||||
|
description: "".to_string(),
|
||||||
|
},
|
||||||
|
Image {
|
||||||
|
path: "bar".into(),
|
||||||
|
description: "".to_string(),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
children: vec![],
|
||||||
|
};
|
||||||
|
// A child album with some images
|
||||||
|
ad.children.push(AlbumDir {
|
||||||
|
path: "subdir".into(),
|
||||||
|
description: "".to_string(),
|
||||||
|
images: vec![
|
||||||
|
Image {
|
||||||
|
path: "subdir/foo".into(),
|
||||||
|
description: "".to_string(),
|
||||||
|
},
|
||||||
|
Image {
|
||||||
|
path: "subdir/bar".into(),
|
||||||
|
description: "".to_string(),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
children: vec![AlbumDir {
|
||||||
|
path: "deeper_subdir".into(),
|
||||||
|
description: "".to_string(),
|
||||||
|
images: vec![Image {
|
||||||
|
path: "deeper_subdir/image.jpg".into(),
|
||||||
|
description: "".to_string(),
|
||||||
|
}],
|
||||||
|
children: vec![],
|
||||||
|
}],
|
||||||
|
});
|
||||||
|
// A child album with no images
|
||||||
|
ad.children.push(AlbumDir {
|
||||||
|
description: "".to_string(),
|
||||||
|
path: "another_subdir".into(),
|
||||||
|
images: vec![],
|
||||||
|
children: vec![],
|
||||||
|
});
|
||||||
|
|
||||||
|
let imgs: HashSet<String> = ad
|
||||||
|
.iter()
|
||||||
|
.map(|i| i.path.clone().to_str().unwrap().to_string())
|
||||||
|
.collect();
|
||||||
|
let expected: HashSet<String> = [
|
||||||
|
"foo",
|
||||||
|
"bar",
|
||||||
|
"subdir/foo",
|
||||||
|
"subdir/bar",
|
||||||
|
"deeper_subdir/image.jpg",
|
||||||
|
]
|
||||||
|
.iter()
|
||||||
|
.map(|s| s.to_string())
|
||||||
|
.collect();
|
||||||
|
assert_eq!(imgs, expected);
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,4 +1,5 @@
|
||||||
use clap::{Parser, Subcommand};
|
use clap::{Parser, Subcommand};
|
||||||
|
use photojawn::generate::generate;
|
||||||
use photojawn::skel::make_skeleton;
|
use photojawn::skel::make_skeleton;
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
|
||||||
|
@ -14,7 +15,7 @@ fn main() -> anyhow::Result<()> {
|
||||||
}
|
}
|
||||||
Commands::Generate { quick } => {
|
Commands::Generate { quick } => {
|
||||||
println!("Generate, quick: {quick}");
|
println!("Generate, quick: {quick}");
|
||||||
todo!()
|
generate(&album_path.to_path_buf());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -20,8 +20,8 @@ pub fn make_skeleton(album_path: &Path) -> Result<(), InitError> {
|
||||||
include_bytes!("../resources/skel/photojawn.conf.yml").as_slice(),
|
include_bytes!("../resources/skel/photojawn.conf.yml").as_slice(),
|
||||||
),
|
),
|
||||||
(
|
(
|
||||||
album_path.join("static/index.css"),
|
album_path.join("_static/index.css"),
|
||||||
include_bytes!("../resources/skel/static/index.css").as_slice(),
|
include_bytes!("../resources/skel/_static/index.css").as_slice(),
|
||||||
),
|
),
|
||||||
(
|
(
|
||||||
album_path.join("_templates/base.html"),
|
album_path.join("_templates/base.html"),
|
||||||
|
|
Loading…
Add table
Reference in a new issue