2024-03-28 23:06:36 +03:00
|
|
|
use std::{
|
|
|
|
fs::{create_dir, File, OpenOptions},
|
|
|
|
io::{self, Write},
|
|
|
|
path::Path,
|
|
|
|
};
|
|
|
|
|
|
|
|
pub static EMBEDDED_FILES: EmbeddedFiles = rustlings_macros::include_files!();
|
|
|
|
|
|
|
|
#[derive(Clone, Copy)]
|
|
|
|
pub enum WriteStrategy {
|
|
|
|
IfNotExists,
|
|
|
|
Overwrite,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl WriteStrategy {
|
|
|
|
fn open<P: AsRef<Path>>(self, path: P) -> io::Result<File> {
|
|
|
|
match self {
|
|
|
|
Self::IfNotExists => OpenOptions::new().create_new(true).write(true).open(path),
|
|
|
|
Self::Overwrite => OpenOptions::new()
|
|
|
|
.create(true)
|
|
|
|
.write(true)
|
|
|
|
.truncate(true)
|
|
|
|
.open(path),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct EmbeddedFile {
|
|
|
|
path: &'static str,
|
|
|
|
content: &'static [u8],
|
|
|
|
}
|
|
|
|
|
|
|
|
impl EmbeddedFile {
|
|
|
|
fn write_to_disk(&self, strategy: WriteStrategy) -> io::Result<()> {
|
|
|
|
strategy.open(self.path)?.write_all(self.content)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct EmbeddedFlatDir {
|
|
|
|
path: &'static str,
|
|
|
|
readme: EmbeddedFile,
|
|
|
|
content: &'static [EmbeddedFile],
|
|
|
|
}
|
|
|
|
|
|
|
|
impl EmbeddedFlatDir {
|
|
|
|
fn init_on_disk(&self) -> io::Result<()> {
|
|
|
|
let path = Path::new(self.path);
|
|
|
|
|
|
|
|
if let Err(e) = create_dir(path) {
|
2024-04-18 02:49:32 +03:00
|
|
|
if e.kind() != io::ErrorKind::AlreadyExists {
|
2024-03-28 23:06:36 +03:00
|
|
|
return Err(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-18 02:49:32 +03:00
|
|
|
self.readme.write_to_disk(WriteStrategy::Overwrite)
|
2024-03-28 23:06:36 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ExercisesDir {
|
|
|
|
readme: EmbeddedFile,
|
|
|
|
files: &'static [EmbeddedFile],
|
|
|
|
dirs: &'static [EmbeddedFlatDir],
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct EmbeddedFiles {
|
|
|
|
exercises_dir: ExercisesDir,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl EmbeddedFiles {
|
|
|
|
pub fn init_exercises_dir(&self) -> io::Result<()> {
|
|
|
|
create_dir("exercises")?;
|
2024-03-29 00:11:16 +03:00
|
|
|
|
2024-03-28 23:06:36 +03:00
|
|
|
self.exercises_dir
|
|
|
|
.readme
|
2024-03-29 00:11:16 +03:00
|
|
|
.write_to_disk(WriteStrategy::IfNotExists)?;
|
|
|
|
|
|
|
|
for file in self.exercises_dir.files {
|
|
|
|
file.write_to_disk(WriteStrategy::IfNotExists)?;
|
|
|
|
}
|
|
|
|
|
|
|
|
for dir in self.exercises_dir.dirs {
|
|
|
|
dir.init_on_disk()?;
|
|
|
|
|
|
|
|
for file in dir.content {
|
|
|
|
file.write_to_disk(WriteStrategy::IfNotExists)?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
2024-03-28 23:06:36 +03:00
|
|
|
}
|
|
|
|
|
2024-04-14 03:41:19 +03:00
|
|
|
pub fn write_exercise_to_disk<P>(&self, path: P, strategy: WriteStrategy) -> io::Result<()>
|
|
|
|
where
|
|
|
|
P: AsRef<Path>,
|
|
|
|
{
|
|
|
|
let path = path.as_ref();
|
|
|
|
|
2024-03-28 23:06:36 +03:00
|
|
|
if let Some(file) = self
|
|
|
|
.exercises_dir
|
|
|
|
.files
|
|
|
|
.iter()
|
2024-03-31 02:49:19 +03:00
|
|
|
.find(|file| Path::new(file.path) == path)
|
2024-03-28 23:06:36 +03:00
|
|
|
{
|
|
|
|
return file.write_to_disk(strategy);
|
|
|
|
}
|
|
|
|
|
|
|
|
for dir in self.exercises_dir.dirs {
|
2024-03-31 02:49:19 +03:00
|
|
|
if let Some(file) = dir.content.iter().find(|file| Path::new(file.path) == path) {
|
2024-03-28 23:06:36 +03:00
|
|
|
dir.init_on_disk()?;
|
|
|
|
return file.write_to_disk(strategy);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-30 23:15:11 +03:00
|
|
|
Err(io::Error::new(
|
|
|
|
io::ErrorKind::NotFound,
|
|
|
|
format!("{} not found in the embedded files", path.display()),
|
|
|
|
))
|
2024-03-28 23:06:36 +03:00
|
|
|
}
|
|
|
|
}
|