GnoCiYeH 1 year ago
parent
commit
21fc724ca5
11 changed files with 766 additions and 0 deletions
  1. 1 0
      .gitignore
  2. 19 0
      Cargo.toml
  3. 8 0
      src/contants.rs
  4. 75 0
      src/error/mod.rs
  5. 16 0
      src/main.rs
  6. 3 0
      src/parse/mod.rs
  7. 420 0
      src/parse/parse_util/mod.rs
  8. 2 0
      src/types.rs
  9. 136 0
      src/unit/mod.rs
  10. 69 0
      src/unit/service/mod.rs
  11. 17 0
      src/unit/target/mod.rs

+ 1 - 0
.gitignore

@@ -1 +1,2 @@
 /target
+Cargo.lock

+ 19 - 0
Cargo.toml

@@ -0,0 +1,19 @@
+[package]
+name = "DragonReach"
+version = "0.1.0"
+edition = "2021"
+
+[[bin]]
+name = "DragonReach"
+path = "src/main.rs"
+
+# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
+
+[dependencies]
+drstd = {path = "/home/heyicong/DrogonOS-work/dragonos-rs-std"}
+
+[profile.release]
+panic = 'abort'
+
+[profile.dev]
+panic = 'abort'

+ 8 - 0
src/contants.rs

@@ -0,0 +1,8 @@
+pub const AF_INET: i32 = 2;
+pub const AF_INET6: i32 = 10;
+
+pub const IPV4_MIN_MTU: u32 = 68;
+pub const IPV6_MIN_MTU: u32 = 1280;
+
+pub const PRIO_MIN: i8 = -20;
+pub const PRIO_MAX: i8 = 20;

+ 75 - 0
src/error/mod.rs

@@ -0,0 +1,75 @@
+
+#[repr(i32)]
+#[derive(Debug, PartialEq, Eq, Clone)]
+#[allow(dead_code, non_camel_case_types)]
+pub enum SystemError {
+    EPERM = 1,
+    /// 没有指定的文件或目录 No such file or directory.
+    ENOENT = 2,
+    /// 没有这样的进程 No such process.
+    ESRCH = 3,
+    /// 被中断的函数 Interrupted function.
+    EINTR = 4,
+    /// I/O错误 I/O error.
+    EIO = 5,
+    /// 没有这样的设备或地址 No such device or address.
+    ENXIO = 6,
+    /// 参数列表过长,或者在输出buffer中缺少空间 或者参数比系统内建的最大值要大 Argument list too long.
+    E2BIG = 7,
+    /// 可执行文件格式错误 Executable file format error
+    ENOEXEC = 8,
+    /// 错误的文件描述符 Bad file descriptor.
+    EBADF = 9,
+    /// 没有子进程 No child processes.
+    ECHILD = 10,
+    /// 资源不可用,请重试。 Resource unavailable, try again.(may be the same value as [EWOULDBLOCK])
+    ///
+    /// 操作将被禁止 Operation would block.(may be the same value as [EAGAIN]).
+    EAGAIN_OR_EWOULDBLOCK = 11,
+    /// 没有足够的空间 Not enough space.
+    ENOMEM = 12,
+    /// 访问被拒绝 Permission denied
+    EACCES = 13,
+    /// 错误的地址 Bad address
+    EFAULT = 14,
+    /// 需要块设备 Block device required
+    ENOTBLK = 15,
+    /// 设备或资源忙 Device or resource busy.
+    EBUSY = 16,
+    /// 文件已存在 File exists.
+    EEXIST = 17,
+    /// 跨设备连接 Cross-device link.
+    EXDEV = 18,
+    /// 没有指定的设备 No such device.
+    ENODEV = 19,
+    /// 不是目录 Not a directory.
+    ENOTDIR = 20,
+    /// 是一个目录 Is a directory
+    EISDIR = 21,
+    /// 不可用的参数 Invalid argument.
+    EINVAL = 22,
+    /// 系统中打开的文件过多 Too many files open in system.
+    ENFILE = 23,
+    /// 文件描述符的值过大 File descriptor value too large.
+    EMFILE = 24,
+    /// 不正确的I/O控制操作 Inappropriate I/O control operation.
+    ENOTTY = 25,
+    /// 文本文件忙 Text file busy.
+    ETXTBSY = 26,
+    /// 文件太大 File too large.
+    EFBIG = 27,
+    /// 设备上没有空间 No space left on device.
+    ENOSPC = 28,
+    /// 错误的寻道.当前文件是pipe,不允许seek请求  Invalid seek.
+    ESPIPE = 29,
+    /// 只读的文件系统 Read-only file system.
+    EROFS = 30,
+    /// 链接数过多 Too many links.
+    EMLINK = 31,
+    /// 断开的管道 Broken pipe.
+    EPIPE = 32,
+    /// 数学参数超出作用域 Mathematics argument out of domain of function.
+    EDOM = 33,
+    /// 结果过大 Result too large.
+    ERANGE = 34,
+}

+ 16 - 0
src/main.rs

@@ -0,0 +1,16 @@
+#![no_std]
+#![no_main]
+
+extern crate drstd;
+use drstd::std as std;
+
+mod parse;
+mod error;
+mod types;
+mod contants;
+mod unit;
+
+#[no_mangle]
+fn main() {
+    
+}

+ 3 - 0
src/parse/mod.rs

@@ -0,0 +1,3 @@
+mod parse_util;
+
+struct UnitParser;

+ 420 - 0
src/parse/parse_util/mod.rs

@@ -0,0 +1,420 @@
+use drstd::std as std;
+use std::{
+    num,
+    vec,
+    vec::Vec,
+    string::ToString
+};
+use crate::{error::SystemError, types::{pid_t, mode_t}, contants::{AF_INET6, IPV6_MIN_MTU, AF_INET, IPV4_MIN_MTU, PRIO_MAX, PRIO_MIN}};
+
+/// @brief 解析布尔值
+/// 
+/// 将传入的字符串解析为布尔值
+/// "yes","y","1","true","t","on"均可表示true
+/// "no","n","0","false","f","off"均可表示false
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @return 解析成功则返回Ok(解析后的值),否则返回Err
+pub fn parse_boolean(s: &str) -> Result<bool,SystemError> {
+    let t_table: Vec<&str> = vec!["yes","y","1","true","t","on"];
+    let f_table: Vec<&str> = vec!["no","n","0","false","f","off"];
+
+    if t_table.contains(&s) {
+        return Ok(true);
+    }else if f_table.contains(&s) {
+        return Ok(false);
+    }
+    
+    return Err(SystemError::EINVAL);
+}
+
+/// @brief 解析pid
+/// 
+/// 将传入的字符串解析为pid
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @return 解析成功则返回Ok(解析后的值),否则返回Err
+pub fn parse_pid(s: &str) -> Result<pid_t,SystemError>{
+    let s = s.trim();
+    //先使用u64变换
+    let pid_ul = match s.parse::<u64>() {
+        Ok(val) => val,
+        Err(_) => {
+            return Err(SystemError::EINVAL);
+        }
+    };
+    let pid: pid_t = pid_ul as pid_t;
+
+    if (pid as u64) != pid_ul {
+        //如果在从pid_t转换为u64之后与之前不等,则说明发生了截断,返回错误
+        return Err(SystemError::ERANGE);
+    }
+
+    if pid < 0 {
+        //pid小于0不合法
+        return Err(SystemError::EINVAL);
+    }
+
+    return Ok(pid);
+}
+
+/// @brief 解析pid
+/// 
+/// 将传入的字符串解析为mode_t
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @return 解析成功则返回Ok(解析后的值),否则返回Err
+pub fn parse_mode(s: &str) -> Result<mode_t,SystemError> {
+    let s = s.trim();
+    let m = match u32::from_str_radix(s, 8) {
+        Ok(val) => val,
+        Err(_) => {
+            return Err(SystemError::EINVAL);
+        }
+    };
+
+    //如果模式大于权限的最大值则为非法权限,返回错误
+    if m > 0o7777 {
+        return Err(SystemError::ERANGE);
+    }
+
+    return Ok(m);
+}
+
+/// @brief 解析网络接口索引
+/// 
+/// 将传入的字符串解析为网络接口索引具体值
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @return 解析成功则返回Ok(解析后的值),否则返回Err
+pub fn parse_ifindex(s: &str) -> Result<i32,SystemError> {
+    let s = s.trim();
+    let ret: i32 = match s.parse::<i32>(){
+        Ok(val) => val,
+        Err(_) => {
+            return Err(SystemError::EINVAL);
+        }
+    };
+
+    if ret <= 0 {
+        return Err(SystemError::EINVAL);
+    }
+
+    return Ok(ret);
+}
+
+/// @brief 解析最大传输单元(MTU)
+/// 
+/// 将传入的字符串解析为具体值
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @param family 网络地址族
+/// 
+/// @return 解析成功则返回Ok(解析后的值),否则返回Err
+pub fn parse_mtu(s: &str,family: i32) -> Result<u32,SystemError> {
+    let s = s.trim();
+    let mtu = match s.parse::<u64>() {
+        Ok(val) => val,
+        Err(_) => {
+            //针对非法字符出错时
+            return Err(SystemError::EINVAL);
+        }
+    };
+    
+    //针对数据溢出时的报错
+    if mtu > u32::MAX as u64 {
+        return Err(SystemError::ERANGE);
+    }
+
+    let mtu: u32 = mtu as u32;
+
+    let mut min_mtu: u32 = 0;
+    //判断mtu是否合法
+    if family == AF_INET6 {
+        min_mtu = IPV6_MIN_MTU;
+    }else if family == AF_INET {
+        min_mtu = IPV4_MIN_MTU;
+    }else {
+        return Err(SystemError::EINVAL);
+    }
+
+    return Ok(mtu);
+}
+
+#[derive(PartialEq)]
+pub enum SizeBase {
+    IEC,
+    Si
+}
+
+struct BaseMapper<'a> {
+    suffix: &'a str,
+    factor: u64
+}
+
+// IEC表
+const BASE_IEC:[BaseMapper;8] = [
+    BaseMapper{ suffix: "E", factor: 1024u64*1024u64*1024u64*1024u64*1024u64*1024u64 },
+    BaseMapper{ suffix: "P", factor: 1024u64*1024u64*1024u64*1024u64*1024u64 },
+    BaseMapper{ suffix: "T", factor: 1024u64*1024u64*1024u64*1024u64 },
+    BaseMapper{ suffix: "G", factor: 1024u64*1024u64*1024u64 },
+    BaseMapper{ suffix: "M", factor: 1024u64*1024u64 },
+    BaseMapper{ suffix: "K", factor: 1024u64 },
+    BaseMapper{ suffix: "B", factor: 1u64 },
+    BaseMapper{ suffix: "",  factor: 1u64 },
+];
+
+// SI表
+const BASE_SI:[BaseMapper;8] = [
+    BaseMapper{ suffix: "E", factor: 1000u64*1000u64*1000u64*1000u64*1000u64*1000u64 },
+    BaseMapper{ suffix: "P", factor: 1000u64*1000u64*1000u64*1000u64*1000u64 },
+    BaseMapper{ suffix: "T", factor: 1000u64*1000u64*1000u64*1000u64 },
+    BaseMapper{ suffix: "G", factor: 1000u64*1000u64*1000u64 },
+    BaseMapper{ suffix: "M", factor: 1000u64*1000u64 },
+    BaseMapper{ suffix: "K", factor: 1000u64 },
+    BaseMapper{ suffix: "B", factor: 1u64 },
+    BaseMapper{ suffix: "",  factor: 1u64 },
+];
+
+/// @brief 解析Size
+/// 
+/// 将传入的字符串解析为具体的字节数
+/// 可支持IEC二进制后缀,也可支持SI十进制后缀
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @param base 设置为IEC二进制后缀或者SI十进制后缀
+/// 
+/// @return 解析成功则返回Ok(解析后的值),否则返回Err
+pub fn parse_size(s: &str, base: SizeBase) -> Result<u64, SystemError> {
+    let s = s.trim();
+    //将s分解为数字和后缀部分
+    let (number_str,suffix) = match s.find(|c: char| !c.is_digit(10) && c != '.'){
+        Some(mid) => s.split_at(mid),
+        None => (s,"")
+    };
+
+    //获得数字部分的整数和小数部分
+    let (integer, fraction) = match number_str.find(".") {
+        Some(mid) => {
+            let (integer, fraction) = number_str.split_at(mid);
+            let integer = integer.parse::<u64>().unwrap();
+            let fraction = match fraction[1..].parse::<u64>() {
+                Ok(val) => val,
+                Err(_) => {
+                    return Err(SystemError::EINVAL);
+                }
+            };
+            (integer,fraction)
+        }
+        None => (number_str.parse::<u64>().unwrap(),0)
+    };
+
+    //从表中查找到后缀所对应的字节倍数
+    let mut factor: u64 = 0;
+    if base == SizeBase::IEC {
+        factor = match BASE_IEC.iter().find(|mapper|mapper.suffix == suffix) {
+            Some(val) => val.factor,
+            None => {
+                return Err(SystemError::EINVAL);
+            }
+        } 
+    } else if base == SizeBase::Si {
+        factor = match BASE_SI.iter().find(|mapper|mapper.suffix == suffix) {
+            Some(val) => val.factor,
+            None => {
+                return Err(SystemError::EINVAL);
+            }
+        } 
+    }
+    
+    Ok(integer*factor + (fraction*factor)/(10u64.pow(fraction.to_string().len() as u32)))
+}
+
+/// @brief 解析扇区大小
+/// 
+/// 将传入的字符串解析为具体的扇区大小
+/// 若扇区大小小于512或者大于4096,将会返回错误,若扇区大小不为2的幂,返回错误。
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @return 解析成功则返回Ok(解析后的值),否则返回Err
+pub fn parse_sector_size(s: &str) -> Result<u64,SystemError> {
+    let s = s.trim();
+    let size: u64 = match s.parse::<u64>(){
+        Ok(val) => val,
+        Err(_) => {
+            return Err(SystemError::EINVAL);
+        }
+    };
+
+    if size < 512 || size > 4096 {
+        return Err(SystemError::ERANGE);
+    }
+
+    //判断是否为2的幂,如果不是则报错
+    if (size & (size - 1)) != 0 {
+        return Err(SystemError::EINVAL);
+    }  
+
+    return Ok(size);
+}
+
+/// @brief 解析范围
+/// 
+/// 将传入的字符串解析为具体的范围
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @return 解析成功则返回Ok(解析后的值),否则返回Err
+pub fn parse_range(s: &str) -> Result<(u32,u32),SystemError> {
+    let mid = match s.find('-') {
+        Some(val) => val,
+        None =>{
+            //如果字符串中没有'-'符号,则表示一个值,所以范围两端都为该值
+            let s = s.trim();
+            let ret = match s.parse::<u32>() {
+                Ok(val) => val,
+                Err(_) => {
+                    return Err(SystemError::EINVAL);
+                }
+            };
+            return Ok((ret,ret));
+        }
+    };
+
+    //若字符串中存在'-',则分别解析为u32,解析失败则报错
+    let (l,r) = s.split_at(mid);
+
+    let l = l.trim();
+    let l = match l.parse::<u32>(){
+        Ok(val) => val,
+        Err(_) => {
+            return Err(SystemError::EINVAL);
+        }
+    };
+    let r = r.trim();
+    let r = match r.parse::<u32>(){
+        Ok(val) => val,
+        Err(_) => {
+            return Err(SystemError::EINVAL);
+        }
+    };
+
+    return Ok((l,r));
+}
+
+/// @brief 解析文件描述符
+/// 
+/// 将传入的字符串解析为文件描述符fd
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @return 解析成功则返回Ok(解析后的值),否则返回Err
+pub fn parse_fd(s: &str) -> Result<i32,SystemError> {
+    let s = s.trim();
+    let fd = match s.parse::<i32>() {
+        Ok(val) => val,
+        Err(_) => {
+            return Err(SystemError::EINVAL);
+        }
+    };
+
+    if fd < 0 {
+        return Err(SystemError::EBADF);
+    }
+
+    return Ok(fd);
+}
+
+/// @brief 解析nice
+/// 
+/// 将传入的字符串解析为nice
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @return 解析成功则返回Ok(解析后的值),否则返回Err
+pub fn parse_nice(s: &str) -> Result<i8,SystemError> {
+    let s = s.trim();
+    let nice = match s.parse::<i8>() {
+        Ok(val) => val,
+        Err(_) => {
+            return Err(SystemError::EINVAL);
+        }
+    };
+
+    if nice > PRIO_MAX || nice < PRIO_MIN {
+        return Err(SystemError::ERANGE);
+    }
+
+    return Ok(nice);
+}
+
+/// @brief 解析端口号
+/// 
+/// 将传入的字符串解析为端口号
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @return 解析成功则返回Ok(解析后的值),否则返回Err
+pub fn parse_ip_port(s: &str) -> Result<u16,SystemError> {
+    let s = s.trim();
+    let port = match s.parse::<u16>() {
+        Ok(val) => val,
+        Err(_) => {
+            return Err(SystemError::EINVAL);
+        }
+    };
+
+    if port == 0 {
+        return Err(SystemError::EINVAL);
+    }
+
+    return Ok(port);
+}
+
+/// @brief 解析端口范围
+/// 
+/// 将传入的字符串解析为端口范围
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @return 解析成功则返回Ok((u16,u16)),否则返回Err
+pub fn parse_ip_port_range(s: &str) -> Result<(u16,u16),SystemError> {
+    let (l,h) = parse_range(s)?;
+
+    let l = l as u16;
+    let h = h as u16;
+    if l <=0 || l >= 65535 || h <= 0 || h >= 65535 {
+        return Err(SystemError::EINVAL);
+    }
+
+    return Ok((l,h));
+}
+
+/// @brief 解析OOM(Out-of-Memory)分数调整值
+/// 
+/// 将传入的字符串解析为OOM(Out-of-Memory)分数调整值
+/// 
+/// @param s 需解析的字符串
+/// 
+/// @return 解析成功则返回Ok(u32),否则返回Err
+pub fn parse_ip_prefix_length(s: &str) -> Result<u32,SystemError> {
+    let len = match s.parse::<u32>() {
+        Ok(val) => val,
+        Err(_) => {
+            return Err(SystemError::EINVAL);
+        }
+    };
+
+    if len > 128 {
+        return Err(SystemError::ERANGE);
+    }
+
+    return Ok(len);
+}

+ 2 - 0
src/types.rs

@@ -0,0 +1,2 @@
+pub type pid_t = i32;
+pub type mode_t = u32;

+ 136 - 0
src/unit/mod.rs

@@ -0,0 +1,136 @@
+use drstd::std as std;
+use std::string::String;
+use std::vec::Vec;
+use std::boxed::Box;
+use crate::error::SystemError;
+
+mod service;
+mod target;
+
+use self::target::TargetUnit;
+
+//所有可解析的Unit都应该实现该trait
+pub trait Unit {
+    fn parse(path: &str) -> Self where Self: Sized;
+}
+
+enum UnitState {
+    Enabled,
+    Disabled,
+    Static,
+    Masked
+}
+
+enum UnitType {
+    Automount,
+    Device,
+    Mount,
+    Path,
+    Scope,
+    Service,
+    Slice,
+    Snapshot,
+    Socket,
+    Swap,
+    Target,
+    Timer,
+}
+
+//记录unit文件基本信息
+struct BaseUnit {
+    unit_part: UnitPart,
+    install_part: InstallPart,
+    state: UnitState,
+    unit_type: UnitType
+}
+
+// impl Default for BaseUnit {
+//     fn default() -> Self {
+//         // BaseUnit { 
+//         //     unit_part: UnitPart::default(), 
+//         //     install_part: InstallPart::default(), 
+//         //     state: , unit_type: () }
+//     }
+// }
+
+struct Url {
+    protocol: String,
+    host: String,
+    port: u16,
+    path: String,
+    query: Option<String>,
+    fragment: Option<String>,
+}
+
+impl Url {
+    fn new(url_string: &str) -> Result<Self, SystemError> {
+        // 解析 URL 字符串并初始化结构体字段
+        // ...
+
+        // 验证 URL 合法性
+        // ...
+
+        // 返回 URL 结构体实例
+        // ...
+        Err(SystemError::EINVAL)
+    }
+
+    fn build(&self) -> String {
+        // 构建 URL 字符串
+        // ...
+        String::new()
+    }
+}
+
+struct UnitPart {
+    description: String,
+    documentation: Vec<Url>,
+    requires: Vec<Box<dyn Unit>>,
+    wants: Vec<Box<dyn Unit>>,
+    after: Vec<Box<dyn Unit>>,
+    before: Vec<Box<dyn Unit>>,
+    binds_to: Vec<Box<dyn Unit>>,
+    part_of: Vec<Box<dyn Unit>>,
+    on_failure: Vec<Box<dyn Unit>>,
+    conflicts: Vec<Box<dyn Unit>>
+}
+
+impl Default for UnitPart {
+    fn default() -> Self {
+        UnitPart { 
+            description: String::new(), 
+            documentation: Vec::new(), 
+            requires: Vec::new(), 
+            wants: Vec::new(), 
+            after: Vec::new(), 
+            before: Vec::new(), 
+            binds_to: Vec::new(), 
+            part_of: Vec::new(), 
+            on_failure: Vec::new(), 
+            conflicts: Vec::new() 
+        }
+    }
+}
+
+struct InstallPart {
+    wanted_by: Vec<TargetUnit>,
+    requires_by: Vec<TargetUnit>,
+    also: Vec<Box<dyn Unit>>,
+    alias: String
+}
+
+impl Default for InstallPart {
+    fn default() -> Self {
+        InstallPart { 
+            wanted_by: Vec::new(), 
+            requires_by: Vec::new(), 
+            also: Vec::new(), 
+            alias: String::new() 
+        }
+    }
+}
+
+struct CmdTask{
+    path: String,
+    cmd: String
+}

+ 69 - 0
src/unit/service/mod.rs

@@ -0,0 +1,69 @@
+use super::{Unit, UnitPart, BaseUnit};
+use drstd::std as std;
+use std::string::String;
+use std::vec::Vec;
+use crate::error::SystemError;
+
+use crate::unit::CmdTask;
+
+struct ServiceUnit{
+    unit_base: BaseUnit,
+    service_part: ServicePart
+}
+
+enum ServiceType {
+    Simple,
+    Forking,
+    OneShot,
+    Dbus,
+    Notify,
+    Idle
+}
+
+enum RestartOption {
+    AlwaysRestart,
+    OnSuccess,
+    OnFailure,
+    OnAbnormal,
+    OnAbort,
+    OnWatchdog
+}
+
+enum MountFlag {
+    Shared,
+    Slave,
+    Private
+}
+
+struct ServicePart {
+    //生命周期相关
+    service_type: ServiceType,
+    /// 
+    remain_after_exit: bool,
+    exec_start: Vec<CmdTask>,
+    exec_start_pre: Vec<CmdTask>,
+    exec_start_pos: Vec<CmdTask>,
+    exec_reload: Vec<CmdTask>,
+    exec_stop: Vec<CmdTask>,
+    exec_stop_post: Vec<CmdTask>,
+    restart_sec: u32,
+    restart: RestartOption,
+    timeout_start_sec: u32,
+    timeout_stop_sec: u32,
+    //上下文配置相关
+    environment: String,
+    environment_file: String,
+    nice: i8,
+    working_directory: String,
+    root_directory: String,
+    user: String,
+    group: String,
+    mount_flags: MountFlag,
+    //LimitCPU / LimitSTACK / LimitNOFILE / LimitNPROC 等,后续支持再添加
+}
+
+impl Unit for ServiceUnit {
+    fn parse(path: &str) -> Self {
+        
+    }
+}

+ 17 - 0
src/unit/target/mod.rs

@@ -0,0 +1,17 @@
+use drstd::std as std;
+use std::vec::Vec;
+use std::boxed::Box;
+
+use super::{Unit, BaseUnit};
+
+
+pub struct TargetUnit {
+    unit_base: BaseUnit,
+    targets: Vec<Box<dyn Unit>>
+}
+
+impl Unit for TargetUnit {
+    fn parse(path: &str) -> Self {
+        
+    }
+}