Split hook/mod.rs into several files
This commit is contained in:
parent
3f15fd0fa8
commit
fa9e8e8192
10 changed files with 260 additions and 263 deletions
21
libimagstore/src/hook/accessor.rs
Normal file
21
libimagstore/src/hook/accessor.rs
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
use hook::result::HookResult;
|
||||||
|
use store::FileLockEntry;
|
||||||
|
|
||||||
|
pub trait MutableHookDataAccessor : Send + Sync {
|
||||||
|
fn access_mut(&self, &mut FileLockEntry) -> HookResult<()>;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait NonMutableHookDataAccessor : Send + Sync {
|
||||||
|
fn access(&self, &FileLockEntry) -> HookResult<()>;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub enum HookDataAccessor {
|
||||||
|
MutableAccess(Box<MutableHookDataAccessor>),
|
||||||
|
NonMutableAccess(Box<NonMutableHookDataAccessor>),
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait HookDataAccessorProvider {
|
||||||
|
fn accessor(&self) -> Box<HookDataAccessor>;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
13
libimagstore/src/hook/create.rs
Normal file
13
libimagstore/src/hook/create.rs
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
use storeid::StoreId;
|
||||||
|
use store::FileLockEntry;
|
||||||
|
use hook::accessor::HookDataAccessorProvider;
|
||||||
|
use hook::result::HookResult;
|
||||||
|
use hook::Hook;
|
||||||
|
|
||||||
|
pub trait PreCreateHook : Hook {
|
||||||
|
fn pre_create(&self, &StoreId) -> HookResult<()>;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait PostCreateHook : Hook + HookDataAccessorProvider {
|
||||||
|
}
|
||||||
|
|
14
libimagstore/src/hook/delete.rs
Normal file
14
libimagstore/src/hook/delete.rs
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
use storeid::StoreId;
|
||||||
|
use store::FileLockEntry;
|
||||||
|
use hook::accessor::HookDataAccessorProvider;
|
||||||
|
use hook::result::HookResult;
|
||||||
|
use hook::Hook;
|
||||||
|
|
||||||
|
pub trait PreDeleteHook : Hook {
|
||||||
|
fn pre_delete(&self, &StoreId) -> HookResult<()>;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait PostDeleteHook : Hook {
|
||||||
|
fn post_delete(&self, &StoreId) -> HookResult<()>;
|
||||||
|
}
|
||||||
|
|
159
libimagstore/src/hook/error.rs
Normal file
159
libimagstore/src/hook/error.rs
Normal file
|
@ -0,0 +1,159 @@
|
||||||
|
use std::error::Error;
|
||||||
|
use std::fmt::Error as FmtError;
|
||||||
|
use std::clone::Clone;
|
||||||
|
use std::fmt::{Display, Formatter};
|
||||||
|
use std::convert::Into;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Kind of error
|
||||||
|
*/
|
||||||
|
#[derive(Clone, Copy, Debug)]
|
||||||
|
pub enum HookErrorKind {
|
||||||
|
Pre(PreHookErrorKind),
|
||||||
|
Post(PostHookErrorKind)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
||||||
|
pub enum PreHookErrorKind {
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Into<HookErrorKind> for PreHookErrorKind {
|
||||||
|
fn into(self) -> HookErrorKind {
|
||||||
|
HookErrorKind::Pre(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
||||||
|
pub enum PostHookErrorKind {
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Into<HookErrorKind> for PostHookErrorKind {
|
||||||
|
fn into(self) -> HookErrorKind {
|
||||||
|
HookErrorKind::Post(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait IntoHookError {
|
||||||
|
fn into_hookerror(self) -> HookError;
|
||||||
|
fn into_hookerror_with_cause(self, cause: Box<Error>) -> HookError;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Into<HookError> for HookErrorKind {
|
||||||
|
|
||||||
|
fn into(self) -> HookError {
|
||||||
|
HookError::new(self, None)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Into<HookError> for (HookErrorKind, Box<Error>) {
|
||||||
|
|
||||||
|
fn into(self) -> HookError {
|
||||||
|
HookError::new(self.0, Some(self.1))
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Into<HookError> for PreHookErrorKind {
|
||||||
|
|
||||||
|
fn into(self) -> HookError {
|
||||||
|
HookError::new(HookErrorKind::Pre(self), None)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Into<HookError> for (PreHookErrorKind, Box<Error>) {
|
||||||
|
|
||||||
|
fn into(self) -> HookError {
|
||||||
|
HookError::new(HookErrorKind::Pre(self.0), Some(self.1))
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Into<HookError> for PostHookErrorKind {
|
||||||
|
|
||||||
|
fn into(self) -> HookError {
|
||||||
|
HookError::new(HookErrorKind::Post(self), None)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Into<HookError> for (PostHookErrorKind, Box<Error>) {
|
||||||
|
|
||||||
|
fn into(self) -> HookError {
|
||||||
|
HookError::new(HookErrorKind::Post(self.0), Some(self.1))
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
fn hook_error_type_as_str(e: &HookErrorKind) -> &'static str {
|
||||||
|
match e {
|
||||||
|
_ => "",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for HookErrorKind {
|
||||||
|
|
||||||
|
fn fmt(&self, fmt: &mut Formatter) -> Result<(), FmtError> {
|
||||||
|
try!(write!(fmt, "{}", hook_error_type_as_str(self)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Error type
|
||||||
|
*/
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct HookError {
|
||||||
|
err_type: HookErrorKind,
|
||||||
|
cause: Option<Box<Error>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl HookError {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build a new HookError from an HookErrorKind, optionally with cause
|
||||||
|
*/
|
||||||
|
pub fn new(errtype: HookErrorKind, cause: Option<Box<Error>>)
|
||||||
|
-> HookError
|
||||||
|
{
|
||||||
|
HookError {
|
||||||
|
err_type: errtype,
|
||||||
|
cause: cause,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the error type of this HookError
|
||||||
|
*/
|
||||||
|
pub fn err_type(&self) -> HookErrorKind {
|
||||||
|
self.err_type.clone()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for HookError {
|
||||||
|
|
||||||
|
fn fmt(&self, fmt: &mut Formatter) -> Result<(), FmtError> {
|
||||||
|
try!(write!(fmt, "[{}]", hook_error_type_as_str(&self.err_type.clone())));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Error for HookError {
|
||||||
|
|
||||||
|
fn description(&self) -> &str {
|
||||||
|
hook_error_type_as_str(&self.err_type.clone())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn cause(&self) -> Option<&Error> {
|
||||||
|
self.cause.as_ref().map(|e| &**e)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,267 +5,16 @@ use toml::Value;
|
||||||
use self::error::HookError;
|
use self::error::HookError;
|
||||||
use store::FileLockEntry;
|
use store::FileLockEntry;
|
||||||
|
|
||||||
pub type HookResult<T> = Result<T, HookError>;
|
pub mod accessor;
|
||||||
|
pub mod create;
|
||||||
|
pub mod delete;
|
||||||
|
pub mod error;
|
||||||
|
pub mod read;
|
||||||
|
pub mod result;
|
||||||
|
pub mod retrieve;
|
||||||
|
pub mod update;
|
||||||
|
|
||||||
pub trait Configureable {
|
pub trait Hook : Debug + Send + Sync {
|
||||||
fn set_config(&mut self, cfg: Value);
|
fn set_config(&mut self, cfg: Value);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait MutableHookDataAccessor : Send + Sync {
|
|
||||||
fn access_mut(&self, &mut FileLockEntry) -> HookResult<()>;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait NonMutableHookDataAccessor : Send + Sync {
|
|
||||||
fn access(&self, &FileLockEntry) -> HookResult<()>;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub enum HookDataAccessor {
|
|
||||||
MutableAccess(Box<MutableHookDataAccessor>),
|
|
||||||
NonMutableAccess(Box<NonMutableHookDataAccessor>),
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait HookDataAccessorProvider {
|
|
||||||
fn accessor(&self) -> Box<HookDataAccessor>;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait Hook : Configureable + Debug + Send + Sync {
|
|
||||||
}
|
|
||||||
|
|
||||||
pub mod read {
|
|
||||||
use storeid::StoreId;
|
|
||||||
use store::FileLockEntry;
|
|
||||||
use super::HookDataAccessorProvider;
|
|
||||||
use super::HookResult;
|
|
||||||
use super::Hook;
|
|
||||||
|
|
||||||
pub trait PreReadHook : Hook {
|
|
||||||
fn pre_read(&self, &StoreId) -> HookResult<()>;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait PostReadHook : Hook + HookDataAccessorProvider {
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
pub mod create {
|
|
||||||
use storeid::StoreId;
|
|
||||||
use store::FileLockEntry;
|
|
||||||
use super::HookDataAccessorProvider;
|
|
||||||
use super::HookResult;
|
|
||||||
use super::Hook;
|
|
||||||
|
|
||||||
pub trait PreCreateHook : Hook {
|
|
||||||
fn pre_create(&self, &StoreId) -> HookResult<()>;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait PostCreateHook : Hook + HookDataAccessorProvider {
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
pub mod retrieve {
|
|
||||||
use storeid::StoreId;
|
|
||||||
use store::FileLockEntry;
|
|
||||||
use super::HookDataAccessorProvider;
|
|
||||||
use super::HookResult;
|
|
||||||
use super::Hook;
|
|
||||||
|
|
||||||
pub trait PreRetrieveHook : Hook {
|
|
||||||
fn pre_retrieve(&self, &StoreId) -> HookResult<()>;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait PostRetrieveHook : Hook + HookDataAccessorProvider {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub mod update {
|
|
||||||
use store::FileLockEntry;
|
|
||||||
use super::HookDataAccessorProvider;
|
|
||||||
use super::HookResult;
|
|
||||||
use super::Hook;
|
|
||||||
|
|
||||||
pub trait PreUpdateHook : Hook {
|
|
||||||
fn pre_update(&self, &FileLockEntry) -> HookResult<()>;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait PostUpdateHook : Hook + HookDataAccessorProvider {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub mod delete {
|
|
||||||
use storeid::StoreId;
|
|
||||||
use store::FileLockEntry;
|
|
||||||
use super::HookDataAccessorProvider;
|
|
||||||
use super::HookResult;
|
|
||||||
use super::Hook;
|
|
||||||
|
|
||||||
pub trait PreDeleteHook : Hook {
|
|
||||||
fn pre_delete(&self, &StoreId) -> HookResult<()>;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait PostDeleteHook : Hook {
|
|
||||||
fn post_delete(&self, &StoreId) -> HookResult<()>;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub mod error {
|
|
||||||
use std::error::Error;
|
|
||||||
use std::fmt::Error as FmtError;
|
|
||||||
use std::clone::Clone;
|
|
||||||
use std::fmt::{Display, Formatter};
|
|
||||||
use std::convert::Into;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Kind of error
|
|
||||||
*/
|
|
||||||
#[derive(Clone, Copy, Debug)]
|
|
||||||
pub enum HookErrorKind {
|
|
||||||
Pre(PreHookErrorKind),
|
|
||||||
Post(PostHookErrorKind)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
||||||
pub enum PreHookErrorKind {
|
|
||||||
// ...
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Into<HookErrorKind> for PreHookErrorKind {
|
|
||||||
fn into(self) -> HookErrorKind {
|
|
||||||
HookErrorKind::Pre(self)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
||||||
pub enum PostHookErrorKind {
|
|
||||||
// ...
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Into<HookErrorKind> for PostHookErrorKind {
|
|
||||||
fn into(self) -> HookErrorKind {
|
|
||||||
HookErrorKind::Post(self)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait IntoHookError {
|
|
||||||
fn into_hookerror(self) -> HookError;
|
|
||||||
fn into_hookerror_with_cause(self, cause: Box<Error>) -> HookError;
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Into<HookError> for HookErrorKind {
|
|
||||||
|
|
||||||
fn into(self) -> HookError {
|
|
||||||
HookError::new(self, None)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Into<HookError> for (HookErrorKind, Box<Error>) {
|
|
||||||
|
|
||||||
fn into(self) -> HookError {
|
|
||||||
HookError::new(self.0, Some(self.1))
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Into<HookError> for PreHookErrorKind {
|
|
||||||
|
|
||||||
fn into(self) -> HookError {
|
|
||||||
HookError::new(HookErrorKind::Pre(self), None)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Into<HookError> for (PreHookErrorKind, Box<Error>) {
|
|
||||||
|
|
||||||
fn into(self) -> HookError {
|
|
||||||
HookError::new(HookErrorKind::Pre(self.0), Some(self.1))
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Into<HookError> for PostHookErrorKind {
|
|
||||||
|
|
||||||
fn into(self) -> HookError {
|
|
||||||
HookError::new(HookErrorKind::Post(self), None)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Into<HookError> for (PostHookErrorKind, Box<Error>) {
|
|
||||||
|
|
||||||
fn into(self) -> HookError {
|
|
||||||
HookError::new(HookErrorKind::Post(self.0), Some(self.1))
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
fn hook_error_type_as_str(e: &HookErrorKind) -> &'static str {
|
|
||||||
match e {
|
|
||||||
_ => "",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Display for HookErrorKind {
|
|
||||||
|
|
||||||
fn fmt(&self, fmt: &mut Formatter) -> Result<(), FmtError> {
|
|
||||||
try!(write!(fmt, "{}", hook_error_type_as_str(self)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Error type
|
|
||||||
*/
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct HookError {
|
|
||||||
err_type: HookErrorKind,
|
|
||||||
cause: Option<Box<Error>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl HookError {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Build a new HookError from an HookErrorKind, optionally with cause
|
|
||||||
*/
|
|
||||||
pub fn new(errtype: HookErrorKind, cause: Option<Box<Error>>)
|
|
||||||
-> HookError
|
|
||||||
{
|
|
||||||
HookError {
|
|
||||||
err_type: errtype,
|
|
||||||
cause: cause,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the error type of this HookError
|
|
||||||
*/
|
|
||||||
pub fn err_type(&self) -> HookErrorKind {
|
|
||||||
self.err_type.clone()
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Display for HookError {
|
|
||||||
|
|
||||||
fn fmt(&self, fmt: &mut Formatter) -> Result<(), FmtError> {
|
|
||||||
try!(write!(fmt, "[{}]", hook_error_type_as_str(&self.err_type.clone())));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Error for HookError {
|
|
||||||
|
|
||||||
fn description(&self) -> &str {
|
|
||||||
hook_error_type_as_str(&self.err_type.clone())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn cause(&self) -> Option<&Error> {
|
|
||||||
self.cause.as_ref().map(|e| &**e)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
13
libimagstore/src/hook/read.rs
Normal file
13
libimagstore/src/hook/read.rs
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
use storeid::StoreId;
|
||||||
|
use store::FileLockEntry;
|
||||||
|
use hook::accessor::HookDataAccessorProvider;
|
||||||
|
use hook::result::HookResult;
|
||||||
|
use hook::Hook;
|
||||||
|
|
||||||
|
pub trait PreReadHook : Hook {
|
||||||
|
fn pre_read(&self, &StoreId) -> HookResult<()>;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait PostReadHook : Hook + HookDataAccessorProvider {
|
||||||
|
}
|
||||||
|
|
3
libimagstore/src/hook/result.rs
Normal file
3
libimagstore/src/hook/result.rs
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
use hook::error::HookError;
|
||||||
|
|
||||||
|
pub type HookResult<T> = Result<T, HookError>;
|
13
libimagstore/src/hook/retrieve.rs
Normal file
13
libimagstore/src/hook/retrieve.rs
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
use storeid::StoreId;
|
||||||
|
use store::FileLockEntry;
|
||||||
|
use hook::accessor::HookDataAccessorProvider;
|
||||||
|
use hook::result::HookResult;
|
||||||
|
use hook::Hook;
|
||||||
|
|
||||||
|
pub trait PreRetrieveHook : Hook {
|
||||||
|
fn pre_retrieve(&self, &StoreId) -> HookResult<()>;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait PostRetrieveHook : Hook + HookDataAccessorProvider {
|
||||||
|
}
|
||||||
|
|
12
libimagstore/src/hook/update.rs
Normal file
12
libimagstore/src/hook/update.rs
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
use store::FileLockEntry;
|
||||||
|
use hook::accessor::HookDataAccessorProvider;
|
||||||
|
use hook::result::HookResult;
|
||||||
|
use hook::Hook;
|
||||||
|
|
||||||
|
pub trait PreUpdateHook : Hook {
|
||||||
|
fn pre_update(&self, &FileLockEntry) -> HookResult<()>;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait PostUpdateHook : Hook + HookDataAccessorProvider {
|
||||||
|
}
|
||||||
|
|
|
@ -23,8 +23,8 @@ use error::{StoreError, StoreErrorKind};
|
||||||
use storeid::{StoreId, StoreIdIterator};
|
use storeid::{StoreId, StoreIdIterator};
|
||||||
use lazyfile::LazyFile;
|
use lazyfile::LazyFile;
|
||||||
|
|
||||||
use hook::HookResult;
|
use hook::result::HookResult;
|
||||||
use hook::{ MutableHookDataAccessor,
|
use hook::accessor::{ MutableHookDataAccessor,
|
||||||
NonMutableHookDataAccessor,
|
NonMutableHookDataAccessor,
|
||||||
HookDataAccessor,
|
HookDataAccessor,
|
||||||
HookDataAccessorProvider};
|
HookDataAccessorProvider};
|
||||||
|
@ -458,7 +458,7 @@ impl Store {
|
||||||
use std::thread;
|
use std::thread;
|
||||||
use std::thread::JoinHandle;
|
use std::thread::JoinHandle;
|
||||||
|
|
||||||
use hook::HookDataAccessor as HDA;
|
use hook::accessor::HookDataAccessor as HDA;
|
||||||
use error::StoreError as SE;
|
use error::StoreError as SE;
|
||||||
use error::StoreErrorKind as SEK;
|
use error::StoreErrorKind as SEK;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue