tinySQL  0.1
A self-contained database management system
Catalog_Manager.cpp
1 #include <iostream>
2 #include "catalog/catalog.h"
3 
4 CatalogManager::CatalogManager(BufferManager *bfm):_bfm(bfm){
5  //創建一個紀錄有多少table的file
6  pageId_t p_Id;
7  if (!_bfm->FileExists(ALL_TABLE_PATH)){
8  _bfm->createEmptyFile(ALL_TABLE_PATH);
9  }
10 
11  char* buffer = _bfm->getPage(ALL_TABLE_PATH, 0, p_Id);
12  _bfm->modifyPage(p_Id);
13  std::string str_buffer = buffer;
14  std::string star = "*";
15  std::string::size_type idx;
16  idx = str_buffer.find(star);
17  if(idx == std::string::npos) // Empty file
18  {
19  strcat(buffer, "*");
20  }
21 
22  int a = -1, i = 0;
23  while(buffer[i]!='*'){
24  if(buffer[i]=='@'){
25  if (i == 0) {
26  a = i;
27  ++i;
28  continue;
29  }
30  tableNames.emplace(str_buffer.substr(a + 1, i - a - 1));
31  a = i;
32  }
33  ++i;
34  }
35  if (i != 0)
36  tableNames.emplace(str_buffer.substr(a + 1, i - a - 1));
37 
38  _bfm->flushPage(p_Id);
39  // buffer save
40 
41  //map table -> has_index
42  for(const auto& name : tableNames){
43  auto index_cat = getIndex(name);
44  for (size_t j = 0; j < index_cat.number; ++j){
45  indexName2tableName.emplace(index_cat.index_name[j], name);
46  }
47  }
48 }
49 
50 CatalogManager::~CatalogManager(){
51  // Serialize all table info to the file
52  std::string allTableInfo;
53  for (auto name : tableNames){
54  allTableInfo += "@" + name;
55  }
56  allTableInfo += "*";
57 
58  pageId_t alltable_pId;
59  char *allTableFile = _bfm->getPage(ALL_TABLE_PATH, 0, alltable_pId);
60  _bfm->modifyPage(alltable_pId);
61 
62  strcpy(allTableFile, allTableInfo.c_str());
63 }
64 
65 bool CatalogManager::existTable(const std::string& table_name){
66  auto iter = tableNames.find(table_name);
67  return (iter != tableNames.end());
68 }
69 
70 bool CatalogManager::existAttribute(const std::string& table_name, const std::string& attr_name){
71 
72  Attribute attr = getAttribute(table_name);
73  for (int i = 0; i<attr.num; i++){
74  if(attr_name == attr.name[i])
75  return true;
76  }
77 
78  return false;
79 }
80 
81 void CatalogManager::rewriteAttribute(const std::string &table_name, const Attribute &attr, const Index &index)
82 {
83  std::string str;
84  pageId_t buffer_pId;
85  // Just overwrite even if the table file exists.
86  char* buffer = _bfm->getPage(PATH::CATALOG_PATH + table_name, 0, buffer_pId);
87  _bfm->modifyPage(buffer_pId);
88  str = buffer;
89 
90  //信息:"@table_name%attr_num attr_info#primary#index_num index_info\n"
91  std::string attr_index_info_str;
92  attr_index_info_str = std::to_string(attr.num);
93  for (int i=0; i<attr.num; i++){
94  attr_index_info_str += " " + std::to_string((int)attr.type[i]) + " " + attr.name[i] + " " + std::to_string(attr.is_unique[i]);
95  }
96  attr_index_info_str += "#" + std::to_string(attr.primary_Key);
97 
98  //Index info
99  attr_index_info_str += "#" + std::to_string(index.number);
100  for (int i=0; i < index.number; i++){
101  attr_index_info_str +=" " + index.index_name[i] +" "+ std::to_string(index.location[i]);
102  }
103  attr_index_info_str += "\n";
104 
105  auto attr_index_start = str.find_first_of('%');
106  attr_index_start += 1;
107  str.replace(attr_index_start, (str.size() - attr_index_start), attr_index_info_str);
108 
109  strcpy(buffer,str.c_str());
110  _bfm->flushPage(buffer_pId);
111 }
112 
113 void CatalogManager::CreateTable(const std::string& table_name, Attribute &attr){
114  Index index;
115  if(attr.primary_Key >= 0){
116  index.number = 1;
117  index.index_name[0] = table_name + "_" + attr.name[attr.primary_Key];
118  index.location[0] = attr.primary_Key;
119 
120  attr.has_index[attr.primary_Key] = true;
121 
122  // Create the has_index for primary key
123  // TODO : Switch to factory mode
124  IndexManager id_manager(_bfm);
125  id_manager.CreateIndex(index.index_name[0]);
126  }
127  else {
128  index.number = 0;
129  }
130 
131  if(existTable(table_name)){
132  std::cout<<"Error the table has already exist!!!"<<std::endl;
133  throw DB_TABLE_ALREADY_EXIST;
134  }
135  else{
136  tableNames.emplace(table_name);
137  }
138 
139  std::string str;
140  pageId_t buffer_pId;
141  // Just overwrite even if the table file exists.
142  _bfm->createEmptyFile(PATH::CATALOG_PATH + table_name);
143  char* buffer = _bfm->getPage(PATH::CATALOG_PATH + table_name, 0, buffer_pId);
144  _bfm->modifyPage(buffer_pId);
145 
146  //信息:"@table_name%attr_num attr_info#primary#index_num index_info\n"
147  str = "@" + table_name;
148  str = str + "%" + std::to_string(attr.num);
149  for (int i=0; i<attr.num; i++){
150  str += " " + std::to_string((int)attr.type[i]) + " " + attr.name[i] + " " + std::to_string(attr.is_unique[i]);
151  }
152  str = str + "#" + std::to_string(attr.primary_Key);
153  //Index info
154  str = str + "#" + std::to_string(index.number);
155  for (int i=0; i < index.number; i++){
156  str +=" " + index.index_name[i] +" "+ std::to_string(index.location[i]);
157  }
158  str += "\n";
159 
160  strcpy(buffer,str.c_str());
161  _bfm->flushPage(buffer_pId);
162 }
163 
164 void CatalogManager::DropTable(const std::string& table_name){
165  if(!existTable(table_name)){
166  throw;
167  }
168  tableNames.erase(table_name);
169  // Only delete when destruct
170 }
171 
172 void CatalogManager::UpdateIndex(const std::string& table_name, const std::string& attr_name, const std::string& index_name){
173  Index index_record = getIndex(table_name);
174  Attribute attr = getAttribute(table_name);
175  if(!existTable(table_name)){
176  throw DB_TABLE_NOT_EXIST;
177  }
178  if(!existAttribute(table_name, attr_name)){
179  throw DB_COLUMN_NAME_NOT_EXIST;
180  }
181  if(index_record.number>=10){
182  throw DB_FAILED;
183  }
184  for(int i=0; i<index_record.number; i++){
185  if(attr.name[index_record.location[i]] == attr_name){
186  throw DB_INDEX_ALREADY_EXIST;
187  }
188  if(index_record.index_name[i] == index_name){
189  throw DB_INDEX_ALREADY_EXIST;
190  }
191  }
192  //上述無任何的異常后,新增index
193  index_record.index_name[index_record.number] = index_name;
194  for (int i = 0; i<attr.num; i++){
195  if(attr_name == attr.name[i]){
196  //新的index在attribute的那个位置
197  index_record.location[index_record.number] = i;
198  attr.has_index[i] = true;
199  }
200  }
201  ++index_record.number;
202  rewriteAttribute(table_name, attr, index_record);
203 
204 /*
205  // 用參數table_name 和buffer_Manager要 相應的block (getPage)
206  char* buffer = _bfm->getPage(table_name, 0 );
207  // 用Table* a = (Table *)buffer 拿取buffer信息
208  Table* table = (Table*)buffer;
209  // 對a內的index修改, 要用pinPage
210  */
211 }
212 
213 void CatalogManager::DropIndex(const std::string& table_name, const std::string& index_name){
214  Index index_record = getIndex(table_name);
215  Attribute attr = getAttribute(table_name);
216  if(!existTable(table_name)){
217  throw DB_TABLE_NOT_EXIST;
218  }
219  //index位置
220  int number = -1;
221  for (int i = 0; i<index_record.number; i++){
222  if(index_record.index_name[i] == index_name){
223  number = i;
224  break;
225  }
226  }
227  //index不存在
228  if(number == -1){
229  throw DB_INDEX_NOT_FOUND;
230  }
231  //index存在 delete it
232  index_record.number = index_record.number - 1;
233  int theLast = index_record.number;
234  index_record.index_name[number] = index_record.index_name[theLast];
235  index_record.location[number] = index_record.location[theLast];
236 
237  rewriteAttribute(table_name, attr, index_record);
238 }
239 
240 Index CatalogManager::getIndex(const std::string& table_name){
241  if(!existTable(table_name)){
242  throw;
243  }
244  char* buffer = _bfm->getPage(PATH::CATALOG_PATH + table_name,0);
245  Index index_record;
246  std::string table_info = buffer;
247  int current = table_info.find_last_of('#');
248 // table_info.erase(current + 1);
249  table_info.erase(0, current + 1);
250 // table_info = table_info.substr(current);
251  current = 0;
252  while(table_info[current]!=' '){
253  current++;
254  }
255  std::string indexNum;
256  indexNum = table_info.substr(0,current);
257  index_record.number = atoi(indexNum.c_str());
258  table_info.erase(0, current + 1);
259 // table_info = table_info.substr(current+1);
260  if(index_record.number>10){
261  throw;
262  }
263  for(int i = 0; i<index_record.number; i++){
264  current = 0;
265  while(table_info[current]!=' '){
266  current++;
267  }
268  index_record.index_name[i] = table_info.substr(0,current);
269  table_info.erase(0, current + 1);
270 // table_info = table_info.substr(current+1);
271  current = 0;
272  while(table_info[current]!=' '){
273  if(table_info[current]=='\n'){
274  break;
275  }
276  current++;
277  }
278  index_record.location[i] = atoi(table_info.substr(0, current).c_str());
279  table_info.erase(0, current + 1);
280 // table_info = table_info.substr(current+1);
281  }
282  return index_record;
283 }
284 
285 Attribute CatalogManager::getAttribute(const std::string& table_name){
286  if(!existTable(table_name)){
287  throw DB_TABLE_NOT_EXIST;
288  }
289  pageId_t p_id;
290  char* buffer = _bfm->getPage(PATH::CATALOG_PATH + table_name,0, p_id);
291  _bfm->pinPage(p_id);
292 
293  Attribute attr_record;
294  std::string attr_info(buffer);
295  int current = 0;
296  while(attr_info[current] != '%'){
297  current++;
298  }
299 // std::string str = attr_info.substr(current, 1);
300  ++current;
301  attr_info.erase(0, current);
302 // attr_info = attr_info.substr(current);
303  attr_record.num = atoi(attr_info.substr(0, 1).c_str());
304 
305  attr_info.erase(0, 2);
306 // attr_info = attr_info.substr(2); // Go over the attr_num field.
307  for(int i = 0; i < attr_record.num; i++){
308  for(int j = 0; j<2; j++){
309  current = 0 ;
310  while(attr_info[current] != ' '){
311  ++current;
312  }
313  if(j == 0)
314  attr_record.type[i] = (BASE_SQL_ValType)atoi(attr_info.substr(0,current).c_str());
315  else
316  attr_record.name[i] = attr_info.substr(0,current);
317  attr_info.erase(0, current+1);
318 // auto temp = attr_info.substr(current+1, attr_info.size());
319 // attr_info = temp;
320  }
321  current = 0;
322  while(attr_info[current]!=' '){
323  if(attr_info[current]=='#'){
324  break;
325  }
326  ++current;
327  }
328  if(attr_info.substr(0,current) == "1")
329  attr_record.is_unique[i] = true;
330  else
331  attr_record.is_unique[i] = false;
332 
333  attr_info.erase(0, current+1);
334  }
335 
336  // primary key info
337  current = 0;
338  while(attr_info[current]!='#'){
339  current++;
340  }
341  attr_record.primary_Key = atoi(attr_info.substr(0,current).c_str());
342 
343  // has_index info
344  Index index_record = getIndex(table_name);
345  for(int i=0; i<index_record.number; i++)
346  attr_record.has_index[index_record.location[i]] = true;
347 
348  _bfm->unpinPage(p_id);
349 
350  return attr_record;
351 }
352 
353 std::string CatalogManager::Index2Attr(const std::string& table_name, const std::string& attr_name, const std::string& index_name){
354  if(!existTable(table_name)){
355  throw;
356  }
357  Index index_record = getIndex(table_name);
358  int found = -1;
359  for (int i = 0; i<index_record.number; i++){
360  if(index_record.index_name[i] == index_name){
361  found = i;
362  break;
363  }
364  }
365  if(found == -1){
366  throw DB_INDEX_NOT_FOUND;
367  }
368  Attribute attr_record = getAttribute(table_name);
369  return attr_record.name[index_record.location[found]];
370 }
371 
372 void CatalogManager::ShowTable(const std::string& table_name){
373  if(!existTable(table_name)){
374  throw;
375  }
376  std::cout<<"Table name:"<<table_name<<std::endl;
377  Attribute attr_record=getAttribute(table_name);
378  Index index_record=getIndex(table_name);
379 
380  std::string attr_type;
381  std::cout<<"Attribute number:"<<attr_record.num<<std::endl;
382  std::cout << "########\n";
383  for(int i = 0; i<attr_record.num; i++){
384  switch (attr_record.type[i]){
385  case BASE_SQL_ValType::INT:
386  attr_type = "int";
387  break;
388  case BASE_SQL_ValType::FLOAT:
389  attr_type = "float";
390  break;
391  default:
392  attr_type = "std::string";
393  break;
394 
395  }
396  std::cout<<"Attr_type : ["<<attr_type << "]\n";
397  std::cout<<"Attr_name : ["<<attr_record.name[i]<<"]\n";
398  std::cout << "Attr_unique : [" << (attr_record.is_unique[i] ? "Yes" : "No") << "]\n";
399  if(i == attr_record.primary_Key){
400  std::cout<<"Primary Key"<<std::endl;
401  }
402  std::cout << "#######\n";
403  }
404  std::cout<<"Index number : ["<<index_record.number<<"]\n";
405  for (int i = 0; i<index_record.number; i++){
406  std::cout<<"Index name : ["<<index_record.index_name[i]<<"]\n";
407  std::cout<<"Index location : ["<<index_record.location[i]<< "]\n";
408  }
409 }
410 
411 std::string CatalogManager::getIndexName(const std::string& table_name, const std::string& attr_name){
412  Index index = getIndex(table_name);
413  Attribute attr = getAttribute(table_name);
414  int location;
415  for(int i = 0; i < attr.num; i++)
416  {
417  if(attr_name == attr.name[i]){
418  location = i;
419  break;
420  }
421  }
422  for(int j = 0; j < 10; j++)
423  {
424  if(location == index.location[j])
425  {
426  return index.index_name[j];
427  }
428  }
429  throw DB_INDEX_NOT_FOUND;
430 }
431 
432 void CatalogManager::ShowAllTable() {
433  std::cout << "----------------------------------------------\n";
434  for (const auto & name : tableNames){
435  ShowTable(name);
436  std::cout << "-------------------------------------------------\n";
437  }
438 }
Attribute::name
std::string name[32]
property name
Definition: data_t.h:175
CatalogManager::existTable
bool existTable(const std::string &table_name)
Judge if the table exists.
Definition: Catalog_Manager.cpp:65
Index::location
int location[10]
where it is in Attribute
Definition: data_t.h:187
CatalogManager::getIndex
Index getIndex(const std::string &table_name)
Get the Index object from table name.
Definition: Catalog_Manager.cpp:240
BufferManager::unpinPage
int unpinPage(int page_id)
解钉一次
Definition: buffer_manager.cpp:173
Index::index_name
std::string index_name[10]
has_index name
Definition: data_t.h:188
BufferManager
Buffer manager is an abstraction of memory on computer for modules at higher level.
Definition: buffer_manager.h:69
CatalogManager::UpdateIndex
void UpdateIndex(const std::string &table_name, const std::string &attr_name, const std::string &index_name)
Update Index.
Definition: Catalog_Manager.cpp:172
CatalogManager::CreateTable
void CreateTable(const std::string &table_name, Attribute &attr)
Create a Table object.
Definition: Catalog_Manager.cpp:113
Attribute::is_unique
bool is_unique[32]
uniqure or not
Definition: data_t.h:176
CatalogManager::ShowTable
void ShowTable(const std::string &table_name)
Definition: Catalog_Manager.cpp:372
Attribute::primary_Key
int primary_Key
-1 not exist, 1-32 exist and the place where the primary key are
Definition: data_t.h:178
CatalogManager::Index2Attr
std::string Index2Attr(const std::string &table_name, const std::string &attr_name, const std::string &index_name)
Definition: Catalog_Manager.cpp:353
IndexManager
Manage index files.
Definition: Index.h:15
BufferManager::flushPage
int flushPage(pageId_t page_id)
核心函数之一。内存和磁盘交互的接口。
Definition: buffer_manager.cpp:227
Attribute::num
int num
number of property
Definition: data_t.h:173
CatalogManager::DropIndex
void DropIndex(const std::string &table_name, const std::string &index_name)
Delete an index file.
Definition: Catalog_Manager.cpp:213
Index::number
int number
number of indexes
Definition: data_t.h:186
BufferManager::modifyPage
void modifyPage(int page_id)
标记页是否被修改
Definition: buffer_manager.cpp:155
Attribute::has_index
bool has_index[32]
index exist or not
Definition: data_t.h:177
CatalogManager::existAttribute
bool existAttribute(const std::string &table_name, const std::string &attr_name)
Judge if an attribute in of a table exists.
Definition: Catalog_Manager.cpp:70
BufferManager::pinPage
void pinPage(int page_id)
钉住一页
Definition: buffer_manager.cpp:163
BufferManager::getPage
char * getPage(const std::string &file_name, int block_id)
获取一页
Definition: buffer_manager.cpp:130
Attribute
The attributes for a schema.
Definition: data_t.h:172
CatalogManager::getAttribute
Attribute getAttribute(const std::string &table_name)
Get the Attribute from table name.
Definition: Catalog_Manager.cpp:285
CatalogManager::DropTable
void DropTable(const std::string &table_name)
Delete a table.
Definition: Catalog_Manager.cpp:164
IndexManager::CreateIndex
void CreateIndex(const std::string &indexName)
Create an Index of an attr. Actually creates an has_index file with a head block. File name conventio...
Definition: index.cpp:8
Index
Index desciption for a table.
Definition: data_t.h:185