1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
use std::collections::BTreeMap;
use std::collections::HashMap;
pub use versatile_data::{
Data as Collection
,IdxSized
};
pub use idx_binary::IdxBinary;
mod collection;
pub use collection::CollectionRow;
mod relation;
pub use relation::RelationIndex;
mod session;
pub use session::{
Session
,Record
,UpdateDepends
};
pub struct Database{
root_dir:String
,collections_map:HashMap<String,i32>
,collections:BTreeMap<i32,Collection>
,relation:RelationIndex
}
impl Database{
pub fn new(dir:&str)->Result<Database,std::io::Error>{
let root_dir=if dir.ends_with("/") || dir.ends_with("\\"){
let mut d=dir.to_string();
d.pop();
d
}else{
dir.to_string()
};
Ok(Database{
root_dir:root_dir.to_string()
,collections:BTreeMap::new()
,collections_map:HashMap::new()
,relation:RelationIndex::new(&root_dir)?
})
}
pub fn root_dir(&self)->&str{
&self.root_dir
}
pub fn session<'a>(&'a mut self,name:&'a str)->Result<Session<'a>,std::io::Error>{
Session::new(self,name)
}
fn collection_by_name_or_create(&mut self,name:&str)->Result<i32,std::io::Error>{
let mut max_id=0;
let collections_dir=self.root_dir.to_string()+"/collection/";
if let Ok(dir)=std::fs::read_dir(&collections_dir){
for d in dir.into_iter(){
let d=d.unwrap();
let dt=d.file_type().unwrap();
if dt.is_dir(){
if let Some(fname)=d.path().file_name(){
if let Some(fname)=fname.to_str(){
let s: Vec<&str>=fname.split("_").collect();
if s.len()>1{
if let Ok(i)=s[0].parse(){
max_id=std::cmp::max(max_id,i);
}
if s[1]==name{
if let Some(path)=d.path().to_str(){
let data=Collection::new(path)?;
self.collections_map.insert(name.to_string(),max_id);
self.collections.insert(
max_id
,data
);
return Ok(max_id);
}
}
}
}
}
}
}
}
let collection_id=max_id+1;
let data=Collection::new(&(collections_dir+"/"+&collection_id.to_string()+"_"+name))?;
self.collections_map.insert(name.to_string(),collection_id);
self.collections.insert(
collection_id
,data
);
Ok(collection_id)
}
pub fn collection(&self,id:i32)->Option<&Collection>{
self.collections.get(&id)
}
pub fn collection_mut(&mut self,id:i32)->Option<&mut Collection>{
self.collections.get_mut(&id)
}
pub fn collection_id(&mut self,name:&str)->Result<i32,std::io::Error>{
if self.collections_map.contains_key(name){
Ok(*self.collections_map.get(name).unwrap())
}else{
self.collection_by_name_or_create(name)
}
}
pub fn relation(&self)->&RelationIndex{
&self.relation
}
}