forked from Proxmox/proxmox
apt: avoid direct impl on api types (use traits instead)
So that we can use api types from expternal crate proxmox-apt-api-types. Signed-off-by: Dietmar Maurer <dietmar@proxmox.com>
This commit is contained in:
parent
960131925e
commit
e56e39185a
@ -9,6 +9,8 @@ use crate::repositories::repository::{
|
|||||||
APTRepository, APTRepositoryFileType, APTRepositoryPackageType,
|
APTRepository, APTRepositoryFileType, APTRepositoryPackageType,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use crate::repositories::repository::APTRepositoryImpl;
|
||||||
|
|
||||||
use proxmox_schema::api;
|
use proxmox_schema::api;
|
||||||
|
|
||||||
mod list_parser;
|
mod list_parser;
|
||||||
@ -116,13 +118,46 @@ pub struct APTRepositoryInfo {
|
|||||||
pub message: String,
|
pub message: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl APTRepositoryFile {
|
pub trait APTRepositoryFileImpl {
|
||||||
/// Creates a new `APTRepositoryFile` without parsing.
|
/// Creates a new `APTRepositoryFile` without parsing.
|
||||||
///
|
///
|
||||||
/// If the file is hidden, the path points to a directory, or the extension
|
/// If the file is hidden, the path points to a directory, or the extension
|
||||||
/// is usually ignored by APT (e.g. `.orig`), `Ok(None)` is returned, while
|
/// is usually ignored by APT (e.g. `.orig`), `Ok(None)` is returned, while
|
||||||
/// invalid file names yield an error.
|
/// invalid file names yield an error.
|
||||||
pub fn new<P: AsRef<Path>>(path: P) -> Result<Option<Self>, APTRepositoryFileError> {
|
fn new<P: AsRef<Path>>(path: P) -> Result<Option<APTRepositoryFile>, APTRepositoryFileError>;
|
||||||
|
|
||||||
|
fn with_content(content: String, content_type: APTRepositoryFileType) -> Self;
|
||||||
|
|
||||||
|
/// Check if the file exists.
|
||||||
|
fn exists(&self) -> bool;
|
||||||
|
|
||||||
|
fn read_with_digest(&self) -> Result<(Vec<u8>, [u8; 32]), APTRepositoryFileError>;
|
||||||
|
|
||||||
|
/// Create an `APTRepositoryFileError`.
|
||||||
|
fn err(&self, error: Error) -> APTRepositoryFileError;
|
||||||
|
|
||||||
|
/// Parses the APT repositories configured in the file on disk, including
|
||||||
|
/// disabled ones.
|
||||||
|
///
|
||||||
|
/// Resets the current repositories and digest, even on failure.
|
||||||
|
fn parse(&mut self) -> Result<(), APTRepositoryFileError>;
|
||||||
|
|
||||||
|
/// Writes the repositories to the file on disk.
|
||||||
|
///
|
||||||
|
/// If a digest is provided, checks that the current content of the file still
|
||||||
|
/// produces the same one.
|
||||||
|
fn write(&self) -> Result<(), APTRepositoryFileError>;
|
||||||
|
|
||||||
|
/// Checks if old or unstable suites are configured and that the Debian security repository
|
||||||
|
/// has the correct suite. Also checks that the `stable` keyword is not used.
|
||||||
|
fn check_suites(&self, current_codename: DebianCodename) -> Vec<APTRepositoryInfo>;
|
||||||
|
|
||||||
|
/// Checks for official URIs.
|
||||||
|
fn check_uris(&self) -> Vec<APTRepositoryInfo>;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl APTRepositoryFileImpl for APTRepositoryFile {
|
||||||
|
fn new<P: AsRef<Path>>(path: P) -> Result<Option<Self>, APTRepositoryFileError> {
|
||||||
let path: PathBuf = path.as_ref().to_path_buf();
|
let path: PathBuf = path.as_ref().to_path_buf();
|
||||||
|
|
||||||
let new_err = |path_string: String, err: &str| APTRepositoryFileError {
|
let new_err = |path_string: String, err: &str| APTRepositoryFileError {
|
||||||
@ -197,7 +232,7 @@ impl APTRepositoryFile {
|
|||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn with_content(content: String, content_type: APTRepositoryFileType) -> Self {
|
fn with_content(content: String, content_type: APTRepositoryFileType) -> Self {
|
||||||
Self {
|
Self {
|
||||||
file_type: content_type,
|
file_type: content_type,
|
||||||
content: Some(content),
|
content: Some(content),
|
||||||
@ -207,8 +242,7 @@ impl APTRepositoryFile {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Check if the file exists.
|
fn exists(&self) -> bool {
|
||||||
pub fn exists(&self) -> bool {
|
|
||||||
if let Some(path) = &self.path {
|
if let Some(path) = &self.path {
|
||||||
PathBuf::from(path).exists()
|
PathBuf::from(path).exists()
|
||||||
} else {
|
} else {
|
||||||
@ -216,7 +250,7 @@ impl APTRepositoryFile {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn read_with_digest(&self) -> Result<(Vec<u8>, [u8; 32]), APTRepositoryFileError> {
|
fn read_with_digest(&self) -> Result<(Vec<u8>, [u8; 32]), APTRepositoryFileError> {
|
||||||
if let Some(path) = &self.path {
|
if let Some(path) = &self.path {
|
||||||
let content = std::fs::read(path).map_err(|err| self.err(format_err!("{}", err)))?;
|
let content = std::fs::read(path).map_err(|err| self.err(format_err!("{}", err)))?;
|
||||||
let digest = openssl::sha::sha256(&content);
|
let digest = openssl::sha::sha256(&content);
|
||||||
@ -233,19 +267,14 @@ impl APTRepositoryFile {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Create an `APTRepositoryFileError`.
|
fn err(&self, error: Error) -> APTRepositoryFileError {
|
||||||
pub fn err(&self, error: Error) -> APTRepositoryFileError {
|
|
||||||
APTRepositoryFileError {
|
APTRepositoryFileError {
|
||||||
path: self.path.clone().unwrap_or_default(),
|
path: self.path.clone().unwrap_or_default(),
|
||||||
error: error.to_string(),
|
error: error.to_string(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Parses the APT repositories configured in the file on disk, including
|
fn parse(&mut self) -> Result<(), APTRepositoryFileError> {
|
||||||
/// disabled ones.
|
|
||||||
///
|
|
||||||
/// Resets the current repositories and digest, even on failure.
|
|
||||||
pub fn parse(&mut self) -> Result<(), APTRepositoryFileError> {
|
|
||||||
self.repositories.clear();
|
self.repositories.clear();
|
||||||
self.digest = None;
|
self.digest = None;
|
||||||
|
|
||||||
@ -269,11 +298,7 @@ impl APTRepositoryFile {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Writes the repositories to the file on disk.
|
fn write(&self) -> Result<(), APTRepositoryFileError> {
|
||||||
///
|
|
||||||
/// If a digest is provided, checks that the current content of the file still
|
|
||||||
/// produces the same one.
|
|
||||||
pub fn write(&self) -> Result<(), APTRepositoryFileError> {
|
|
||||||
let path = match &self.path {
|
let path = match &self.path {
|
||||||
Some(path) => path,
|
Some(path) => path,
|
||||||
None => {
|
None => {
|
||||||
@ -336,9 +361,7 @@ impl APTRepositoryFile {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Checks if old or unstable suites are configured and that the Debian security repository
|
fn check_suites(&self, current_codename: DebianCodename) -> Vec<APTRepositoryInfo> {
|
||||||
/// has the correct suite. Also checks that the `stable` keyword is not used.
|
|
||||||
pub fn check_suites(&self, current_codename: DebianCodename) -> Vec<APTRepositoryInfo> {
|
|
||||||
let mut infos = vec![];
|
let mut infos = vec![];
|
||||||
|
|
||||||
let path = match &self.path {
|
let path = match &self.path {
|
||||||
@ -427,8 +450,7 @@ impl APTRepositoryFile {
|
|||||||
infos
|
infos
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Checks for official URIs.
|
fn check_uris(&self) -> Vec<APTRepositoryInfo> {
|
||||||
pub fn check_uris(&self) -> Vec<APTRepositoryInfo> {
|
|
||||||
let mut infos = vec![];
|
let mut infos = vec![];
|
||||||
|
|
||||||
let path = match &self.path {
|
let path = match &self.path {
|
||||||
|
@ -3,9 +3,9 @@ use std::iter::Iterator;
|
|||||||
|
|
||||||
use anyhow::{bail, format_err, Error};
|
use anyhow::{bail, format_err, Error};
|
||||||
|
|
||||||
use crate::repositories::{APTRepository, APTRepositoryFileType, APTRepositoryOption};
|
|
||||||
|
|
||||||
use super::APTRepositoryParser;
|
use super::APTRepositoryParser;
|
||||||
|
use crate::repositories::APTRepositoryImpl;
|
||||||
|
use crate::repositories::{APTRepository, APTRepositoryFileType, APTRepositoryOption};
|
||||||
|
|
||||||
// TODO convert %-escape characters. Also adapt printing back accordingly,
|
// TODO convert %-escape characters. Also adapt printing back accordingly,
|
||||||
// because at least '%' needs to be re-escaped when printing.
|
// because at least '%' needs to be re-escaped when printing.
|
||||||
|
@ -3,6 +3,7 @@ use std::iter::Iterator;
|
|||||||
|
|
||||||
use anyhow::{bail, Error};
|
use anyhow::{bail, Error};
|
||||||
|
|
||||||
|
use crate::repositories::APTRepositoryImpl;
|
||||||
use crate::repositories::{
|
use crate::repositories::{
|
||||||
APTRepository, APTRepositoryFileType, APTRepositoryOption, APTRepositoryPackageType,
|
APTRepository, APTRepositoryFileType, APTRepositoryOption, APTRepositoryPackageType,
|
||||||
};
|
};
|
||||||
|
@ -4,17 +4,20 @@ use std::path::PathBuf;
|
|||||||
use anyhow::{bail, Error};
|
use anyhow::{bail, Error};
|
||||||
|
|
||||||
mod repository;
|
mod repository;
|
||||||
|
pub use repository::APTRepositoryImpl;
|
||||||
pub use repository::{
|
pub use repository::{
|
||||||
APTRepository, APTRepositoryFileType, APTRepositoryOption, APTRepositoryPackageType,
|
APTRepository, APTRepositoryFileType, APTRepositoryOption, APTRepositoryPackageType,
|
||||||
};
|
};
|
||||||
|
|
||||||
mod file;
|
mod file;
|
||||||
|
pub use file::APTRepositoryFileImpl;
|
||||||
pub use file::{APTRepositoryFile, APTRepositoryFileError, APTRepositoryInfo};
|
pub use file::{APTRepositoryFile, APTRepositoryFileError, APTRepositoryInfo};
|
||||||
|
|
||||||
mod release;
|
mod release;
|
||||||
pub use release::{get_current_release_codename, DebianCodename};
|
pub use release::{get_current_release_codename, DebianCodename};
|
||||||
|
|
||||||
mod standard;
|
mod standard;
|
||||||
|
pub use standard::APTRepositoryHandleImpl;
|
||||||
pub use standard::{APTRepositoryHandle, APTStandardRepository};
|
pub use standard::{APTRepositoryHandle, APTStandardRepository};
|
||||||
|
|
||||||
const APT_SOURCES_LIST_FILENAME: &str = "/etc/apt/sources.list";
|
const APT_SOURCES_LIST_FILENAME: &str = "/etc/apt/sources.list";
|
||||||
|
@ -8,6 +8,7 @@ use serde::{Deserialize, Serialize};
|
|||||||
use proxmox_schema::api;
|
use proxmox_schema::api;
|
||||||
|
|
||||||
use crate::repositories::standard::APTRepositoryHandle;
|
use crate::repositories::standard::APTRepositoryHandle;
|
||||||
|
use crate::repositories::standard::APTRepositoryHandleImpl;
|
||||||
|
|
||||||
#[api]
|
#[api]
|
||||||
#[derive(Debug, Copy, Clone, Serialize, Deserialize, PartialEq, Eq)]
|
#[derive(Debug, Copy, Clone, Serialize, Deserialize, PartialEq, Eq)]
|
||||||
@ -188,9 +189,41 @@ pub struct APTRepository {
|
|||||||
pub enabled: bool,
|
pub enabled: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl APTRepository {
|
pub trait APTRepositoryImpl {
|
||||||
/// Crates an empty repository.
|
/// Crates an empty repository.
|
||||||
pub fn new(file_type: APTRepositoryFileType) -> Self {
|
fn new(file_type: APTRepositoryFileType) -> Self;
|
||||||
|
|
||||||
|
/// Changes the `enabled` flag and makes sure the `Enabled` option for
|
||||||
|
/// `APTRepositoryPackageType::Sources` repositories is updated too.
|
||||||
|
fn set_enabled(&mut self, enabled: bool);
|
||||||
|
|
||||||
|
/// Makes sure that all basic properties of a repository are present and not obviously invalid.
|
||||||
|
fn basic_check(&self) -> Result<(), Error>;
|
||||||
|
|
||||||
|
/// Checks if the repository is the one referenced by the handle.
|
||||||
|
fn is_referenced_repository(
|
||||||
|
&self,
|
||||||
|
handle: APTRepositoryHandle,
|
||||||
|
product: &str,
|
||||||
|
suite: &str,
|
||||||
|
) -> bool;
|
||||||
|
|
||||||
|
/// Guess the origin from the repository's URIs.
|
||||||
|
///
|
||||||
|
/// Intended to be used as a fallback for get_cached_origin.
|
||||||
|
fn origin_from_uris(&self) -> Option<String>;
|
||||||
|
|
||||||
|
/// Get the `Origin:` value from a cached InRelease file.
|
||||||
|
fn get_cached_origin(&self) -> Result<Option<String>, Error>;
|
||||||
|
|
||||||
|
/// Writes a repository in the corresponding format followed by a blank.
|
||||||
|
///
|
||||||
|
/// Expects that `basic_check()` for the repository was successful.
|
||||||
|
fn write(&self, w: &mut dyn Write) -> Result<(), Error>;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl APTRepositoryImpl for APTRepository {
|
||||||
|
fn new(file_type: APTRepositoryFileType) -> Self {
|
||||||
Self {
|
Self {
|
||||||
types: vec![],
|
types: vec![],
|
||||||
uris: vec![],
|
uris: vec![],
|
||||||
@ -203,9 +236,7 @@ impl APTRepository {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Changes the `enabled` flag and makes sure the `Enabled` option for
|
fn set_enabled(&mut self, enabled: bool) {
|
||||||
/// `APTRepositoryPackageType::Sources` repositories is updated too.
|
|
||||||
pub fn set_enabled(&mut self, enabled: bool) {
|
|
||||||
self.enabled = enabled;
|
self.enabled = enabled;
|
||||||
|
|
||||||
if self.file_type == APTRepositoryFileType::Sources {
|
if self.file_type == APTRepositoryFileType::Sources {
|
||||||
@ -226,8 +257,7 @@ impl APTRepository {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Makes sure that all basic properties of a repository are present and not obviously invalid.
|
fn basic_check(&self) -> Result<(), Error> {
|
||||||
pub fn basic_check(&self) -> Result<(), Error> {
|
|
||||||
if self.types.is_empty() {
|
if self.types.is_empty() {
|
||||||
bail!("missing package type(s)");
|
bail!("missing package type(s)");
|
||||||
}
|
}
|
||||||
@ -267,8 +297,7 @@ impl APTRepository {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Checks if the repository is the one referenced by the handle.
|
fn is_referenced_repository(
|
||||||
pub fn is_referenced_repository(
|
|
||||||
&self,
|
&self,
|
||||||
handle: APTRepositoryHandle,
|
handle: APTRepositoryHandle,
|
||||||
product: &str,
|
product: &str,
|
||||||
@ -299,10 +328,7 @@ impl APTRepository {
|
|||||||
&& found_component
|
&& found_component
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Guess the origin from the repository's URIs.
|
fn origin_from_uris(&self) -> Option<String> {
|
||||||
///
|
|
||||||
/// Intended to be used as a fallback for get_cached_origin.
|
|
||||||
pub fn origin_from_uris(&self) -> Option<String> {
|
|
||||||
for uri in self.uris.iter() {
|
for uri in self.uris.iter() {
|
||||||
if let Some(host) = host_from_uri(uri) {
|
if let Some(host) = host_from_uri(uri) {
|
||||||
if host == "proxmox.com" || host.ends_with(".proxmox.com") {
|
if host == "proxmox.com" || host.ends_with(".proxmox.com") {
|
||||||
@ -318,8 +344,7 @@ impl APTRepository {
|
|||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get the `Origin:` value from a cached InRelease file.
|
fn get_cached_origin(&self) -> Result<Option<String>, Error> {
|
||||||
pub fn get_cached_origin(&self) -> Result<Option<String>, Error> {
|
|
||||||
for uri in self.uris.iter() {
|
for uri in self.uris.iter() {
|
||||||
for suite in self.suites.iter() {
|
for suite in self.suites.iter() {
|
||||||
let mut file = release_filename(uri, suite, false);
|
let mut file = release_filename(uri, suite, false);
|
||||||
@ -353,10 +378,7 @@ impl APTRepository {
|
|||||||
Ok(None)
|
Ok(None)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Writes a repository in the corresponding format followed by a blank.
|
fn write(&self, w: &mut dyn Write) -> Result<(), Error> {
|
||||||
///
|
|
||||||
/// Expects that `basic_check()` for the repository was successful.
|
|
||||||
pub fn write(&self, w: &mut dyn Write) -> Result<(), Error> {
|
|
||||||
match self.file_type {
|
match self.file_type {
|
||||||
APTRepositoryFileType::List => write_one_line(self, w),
|
APTRepositoryFileType::List => write_one_line(self, w),
|
||||||
APTRepositoryFileType::Sources => write_stanza(self, w),
|
APTRepositoryFileType::Sources => write_stanza(self, w),
|
||||||
|
@ -111,9 +111,26 @@ impl Display for APTRepositoryHandle {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl APTRepositoryHandle {
|
pub trait APTRepositoryHandleImpl {
|
||||||
/// Get the description for the repository.
|
/// Get the description for the repository.
|
||||||
pub fn description(self) -> String {
|
fn description(self) -> String;
|
||||||
|
/// Get the display name of the repository.
|
||||||
|
fn name(self) -> String;
|
||||||
|
/// Get the standard file path for the repository referenced by the handle.
|
||||||
|
fn path(self, product: &str) -> String;
|
||||||
|
/// Get package type, possible URIs and the component associated with the handle.
|
||||||
|
///
|
||||||
|
/// The first URI is the preferred one.
|
||||||
|
fn info(self, product: &str) -> (APTRepositoryPackageType, Vec<String>, String);
|
||||||
|
/// Get the standard repository referenced by the handle.
|
||||||
|
///
|
||||||
|
/// An URI in the result is not '/'-terminated (under the assumption that no valid
|
||||||
|
/// product name is).
|
||||||
|
fn to_repository(self, product: &str, suite: &str) -> APTRepository;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl APTRepositoryHandleImpl for APTRepositoryHandle {
|
||||||
|
fn description(self) -> String {
|
||||||
match self {
|
match self {
|
||||||
APTRepositoryHandle::Enterprise => {
|
APTRepositoryHandle::Enterprise => {
|
||||||
"This is the default, stable, and recommended repository, available for all \
|
"This is the default, stable, and recommended repository, available for all \
|
||||||
@ -155,8 +172,7 @@ impl APTRepositoryHandle {
|
|||||||
.to_string()
|
.to_string()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get the display name of the repository.
|
fn name(self) -> String {
|
||||||
pub fn name(self) -> String {
|
|
||||||
match self {
|
match self {
|
||||||
APTRepositoryHandle::Enterprise => "Enterprise",
|
APTRepositoryHandle::Enterprise => "Enterprise",
|
||||||
APTRepositoryHandle::NoSubscription => "No-Subscription",
|
APTRepositoryHandle::NoSubscription => "No-Subscription",
|
||||||
@ -171,8 +187,7 @@ impl APTRepositoryHandle {
|
|||||||
.to_string()
|
.to_string()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get the standard file path for the repository referenced by the handle.
|
fn path(self, product: &str) -> String {
|
||||||
pub fn path(self, product: &str) -> String {
|
|
||||||
match self {
|
match self {
|
||||||
APTRepositoryHandle::Enterprise => {
|
APTRepositoryHandle::Enterprise => {
|
||||||
format!("/etc/apt/sources.list.d/{}-enterprise.list", product)
|
format!("/etc/apt/sources.list.d/{}-enterprise.list", product)
|
||||||
@ -188,10 +203,7 @@ impl APTRepositoryHandle {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get package type, possible URIs and the component associated with the handle.
|
fn info(self, product: &str) -> (APTRepositoryPackageType, Vec<String>, String) {
|
||||||
///
|
|
||||||
/// The first URI is the preferred one.
|
|
||||||
pub fn info(self, product: &str) -> (APTRepositoryPackageType, Vec<String>, String) {
|
|
||||||
match self {
|
match self {
|
||||||
APTRepositoryHandle::Enterprise => (
|
APTRepositoryHandle::Enterprise => (
|
||||||
APTRepositoryPackageType::Deb,
|
APTRepositoryPackageType::Deb,
|
||||||
@ -259,11 +271,7 @@ impl APTRepositoryHandle {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get the standard repository referenced by the handle.
|
fn to_repository(self, product: &str, suite: &str) -> APTRepository {
|
||||||
///
|
|
||||||
/// An URI in the result is not '/'-terminated (under the assumption that no valid
|
|
||||||
/// product name is).
|
|
||||||
pub fn to_repository(self, product: &str, suite: &str) -> APTRepository {
|
|
||||||
let (package_type, uris, component) = self.info(product);
|
let (package_type, uris, component) = self.info(product);
|
||||||
|
|
||||||
APTRepository {
|
APTRepository {
|
||||||
|
@ -8,6 +8,9 @@ use proxmox_apt::repositories::{
|
|||||||
check_repositories, get_current_release_codename, standard_repositories, APTRepositoryFile,
|
check_repositories, get_current_release_codename, standard_repositories, APTRepositoryFile,
|
||||||
APTRepositoryHandle, APTRepositoryInfo, APTStandardRepository, DebianCodename,
|
APTRepositoryHandle, APTRepositoryInfo, APTStandardRepository, DebianCodename,
|
||||||
};
|
};
|
||||||
|
use proxmox_apt::repositories::{
|
||||||
|
APTRepositoryFileImpl, APTRepositoryHandleImpl, APTRepositoryImpl,
|
||||||
|
};
|
||||||
|
|
||||||
fn create_clean_directory(path: &PathBuf) -> Result<(), Error> {
|
fn create_clean_directory(path: &PathBuf) -> Result<(), Error> {
|
||||||
match std::fs::remove_dir_all(path) {
|
match std::fs::remove_dir_all(path) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user