Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
oceanbase
miniob
提交
1cfc6080
M
miniob
项目概览
oceanbase
/
miniob
1 年多 前同步成功
通知
74
Star
1521
Fork
537
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
分析
仓库
DevOps
项目成员
Pages
M
miniob
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Pages
分析
分析
仓库分析
DevOps
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
提交
提交
1cfc6080
编写于
7月 08, 2022
作者:
羽飞
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
add index scan operator
上级
cd892de3
变更
15
显示空白变更内容
内联
并排
Showing
15 changed file
with
403 addition
and
158 deletion
+403
-158
deps/common/#defer.h#
deps/common/#defer.h#
+21
-0
deps/common/lang/defer.h
deps/common/lang/defer.h
+24
-0
src/observer/sql/executor/execute_stage.cpp
src/observer/sql/executor/execute_stage.cpp
+137
-147
src/observer/sql/expr/expression.cpp
src/observer/sql/expr/expression.cpp
+15
-1
src/observer/sql/expr/expression.h
src/observer/sql/expr/expression.h
+36
-2
src/observer/sql/expr/tuple.h
src/observer/sql/expr/tuple.h
+3
-3
src/observer/sql/expr/tuple_cell.cpp
src/observer/sql/expr/tuple_cell.cpp
+14
-0
src/observer/sql/expr/tuple_cell.h
src/observer/sql/expr/tuple_cell.h
+23
-0
src/observer/sql/operator/index_scan_operator.cpp
src/observer/sql/operator/index_scan_operator.cpp
+67
-0
src/observer/sql/operator/index_scan_operator.h
src/observer/sql/operator/index_scan_operator.h
+33
-0
src/observer/storage/common/field.cpp
src/observer/storage/common/field.cpp
+0
-0
src/observer/storage/common/field.h
src/observer/storage/common/field.h
+14
-0
src/observer/storage/common/record.cpp
src/observer/storage/common/record.cpp
+0
-4
src/observer/storage/common/table.cpp
src/observer/storage/common/table.cpp
+9
-0
src/observer/storage/common/table.h
src/observer/storage/common/table.h
+7
-1
未找到文件。
deps/common/#defer.h#
0 → 100644
浏览文件 @
1cfc6080
#pragma once
namespace common {
class DeferHelper
{
public:
DeferHelper(std::function<void> &defer) : defer_(defer)
{}
~DeferHelper()
{
defer_();
}
private:
std::function<void> &defer_;
};
} // namespace common
#define DEFER(callback) common::DeferHelper defer_helper_##__LINE__(callback)
deps/common/lang/defer.h
0 → 100644
浏览文件 @
1cfc6080
#pragma once
#include <functional>
namespace
common
{
class
DeferHelper
{
public:
DeferHelper
(
const
std
::
function
<
void
()
>
&
defer
)
:
defer_
(
defer
)
{}
~
DeferHelper
()
{
defer_
();
}
private:
const
std
::
function
<
void
()
>
&
defer_
;
};
}
// namespace common
#define DEFER(callback) common::DeferHelper defer_helper_##__LINE__(callback)
src/observer/sql/executor/execute_stage.cpp
浏览文件 @
1cfc6080
...
...
@@ -19,6 +19,7 @@ See the Mulan PSL v2 for more details. */
#include "common/io/io.h"
#include "common/log/log.h"
#include "common/lang/defer.h"
#include "common/seda/timer_stage.h"
#include "common/lang/string.h"
#include "session/session.h"
...
...
@@ -27,6 +28,7 @@ See the Mulan PSL v2 for more details. */
#include "event/session_event.h"
#include "sql/expr/tuple.h"
#include "sql/operator/table_scan_operator.h"
#include "sql/operator/index_scan_operator.h"
#include "sql/operator/predicate_operator.h"
#include "sql/operator/delete_operator.h"
#include "sql/operator/project_operator.h"
...
...
@@ -35,8 +37,10 @@ See the Mulan PSL v2 for more details. */
#include "sql/stmt/update_stmt.h"
#include "sql/stmt/delete_stmt.h"
#include "sql/stmt/insert_stmt.h"
#include "sql/stmt/filter_stmt.h"
#include "storage/common/table.h"
#include "storage/common/field.h"
#include "storage/index/index.h"
#include "storage/default/default_handler.h"
#include "storage/common/condition_filter.h"
#include "storage/trx/trx.h"
...
...
@@ -249,6 +253,131 @@ void tuple_to_string(std::ostream &os, const Tuple &tuple)
cell
.
to_string
(
os
);
}
}
IndexScanOperator
*
try_to_create_index_scan_operator
(
FilterStmt
*
filter_stmt
)
{
const
std
::
vector
<
FilterUnit
*>
&
filter_units
=
filter_stmt
->
filter_units
();
if
(
filter_units
.
empty
()
)
{
return
nullptr
;
}
// 在所有过滤条件中,找到字段与值做比较的条件,然后判断字段是否可以使用索引
// 如果是多列索引,这里的处理需要更复杂。
// 这里的查找规则是比较简单的,就是尽量找到使用相等比较的索引
// 如果没有就找范围比较的,但是直接排除不等比较的索引查询. (你知道为什么?)
const
FilterUnit
*
better_filter
=
nullptr
;
for
(
const
FilterUnit
*
filter_unit
:
filter_units
)
{
if
(
filter_unit
->
comp
()
==
NOT_EQUAL
)
{
continue
;
}
Expression
*
left
=
filter_unit
->
left
();
Expression
*
right
=
filter_unit
->
right
();
CompOp
comp
=
filter_unit
->
comp
();
if
(
left
->
type
()
==
ExprType
::
FIELD
&&
right
->
type
()
==
ExprType
::
VALUE
)
{
}
else
if
(
left
->
type
()
==
ExprType
::
VALUE
&&
right
->
type
()
==
ExprType
::
FIELD
)
{
std
::
swap
(
left
,
right
);
}
FieldExpr
&
left_field_expr
=
*
(
FieldExpr
*
)
left
;
const
Field
&
field
=
left_field_expr
.
field
();
const
Table
*
table
=
field
.
table
();
Index
*
index
=
table
->
find_index_by_field
(
field
.
field_name
());
if
(
index
!=
nullptr
)
{
if
(
better_filter
==
nullptr
)
{
better_filter
=
filter_unit
;
}
else
if
(
filter_unit
->
comp
()
==
EQUAL_TO
)
{
better_filter
=
filter_unit
;
break
;
}
}
}
if
(
better_filter
==
nullptr
)
{
return
nullptr
;
}
Expression
*
left
=
better_filter
->
left
();
Expression
*
right
=
better_filter
->
right
();
CompOp
comp
=
better_filter
->
comp
();
if
(
left
->
type
()
==
ExprType
::
VALUE
&&
right
->
type
()
==
ExprType
::
FIELD
)
{
std
::
swap
(
left
,
right
);
switch
(
comp
)
{
case
EQUAL_TO
:
{
comp
=
EQUAL_TO
;
}
break
;
case
LESS_EQUAL
:
{
comp
=
GREAT_THAN
;
}
break
;
case
NOT_EQUAL
:
{
comp
=
NOT_EQUAL
;
}
break
;
case
LESS_THAN
:
{
comp
=
GREAT_EQUAL
;
}
break
;
case
GREAT_EQUAL
:
{
comp
=
LESS_THAN
;
}
break
;
case
GREAT_THAN
:
{
comp
=
LESS_EQUAL
;
}
break
;
default:
{
LOG_WARN
(
"should not happen"
);
}
}
}
FieldExpr
&
left_field_expr
=
*
(
FieldExpr
*
)
left
;
const
Field
&
field
=
left_field_expr
.
field
();
const
Table
*
table
=
field
.
table
();
Index
*
index
=
table
->
find_index_by_field
(
field
.
field_name
());
assert
(
index
!=
nullptr
);
ValueExpr
&
right_value_expr
=
*
(
ValueExpr
*
)
right
;
TupleCell
value
;
right_value_expr
.
get_tuple_cell
(
value
);
const
TupleCell
*
left_cell
=
nullptr
;
const
TupleCell
*
right_cell
=
nullptr
;
bool
left_inclusive
=
false
;
bool
right_inclusive
=
false
;
switch
(
comp
)
{
case
EQUAL_TO
:
{
left_cell
=
&
value
;
right_cell
=
&
value
;
left_inclusive
=
true
;
right_inclusive
=
true
;
}
break
;
case
LESS_EQUAL
:
{
left_cell
=
nullptr
;
left_inclusive
=
false
;
right_cell
=
&
value
;
right_inclusive
=
true
;
}
break
;
case
LESS_THAN
:
{
left_cell
=
nullptr
;
left_inclusive
=
false
;
right_cell
=
&
value
;
right_inclusive
=
false
;
}
break
;
case
GREAT_EQUAL
:
{
left_cell
=
&
value
;
left_inclusive
=
true
;
right_cell
=
nullptr
;
right_inclusive
=
false
;
}
break
;
case
GREAT_THAN
:
{
left_cell
=
&
value
;
left_inclusive
=
false
;
right_cell
=
nullptr
;
right_inclusive
=
false
;
}
break
;
default:
{
LOG_WARN
(
"should not happen. comp=%d"
,
comp
);
}
break
;
}
IndexScanOperator
*
oper
=
new
IndexScanOperator
(
table
,
index
,
left_cell
,
left_inclusive
,
right_cell
,
right_inclusive
);
LOG_INFO
(
"use index for scan: %s in table %s"
,
index
->
index_meta
().
name
(),
table
->
name
());
return
oper
;
}
RC
ExecuteStage
::
do_select
(
SQLStageEvent
*
sql_event
)
{
SelectStmt
*
select_stmt
=
(
SelectStmt
*
)(
sql_event
->
stmt
());
...
...
@@ -260,10 +389,15 @@ RC ExecuteStage::do_select(SQLStageEvent *sql_event)
return
rc
;
}
Table
*
table
=
select_stmt
->
tables
().
front
();
TableScanOperator
table_scan_operator
(
table
);
Operator
*
scan_oper
=
try_to_create_index_scan_operator
(
select_stmt
->
filter_stmt
());
if
(
nullptr
==
scan_oper
)
{
scan_oper
=
new
TableScanOperator
(
select_stmt
->
tables
()[
0
]);
}
DEFER
([
&
]
()
{
delete
scan_oper
;});
PredicateOperator
pred_oper
(
select_stmt
->
filter_stmt
());
pred_oper
.
add_child
(
&
table_scan_operato
r
);
pred_oper
.
add_child
(
scan_ope
r
);
ProjectOperator
project_oper
;
project_oper
.
add_child
(
&
pred_oper
);
for
(
const
Field
&
field
:
select_stmt
->
query_fields
())
{
...
...
@@ -301,150 +435,6 @@ RC ExecuteStage::do_select(SQLStageEvent *sql_event)
return
rc
;
}
#if 0
// 这里没有对输入的某些信息做合法性校验,比如查询的列名、where条件中的列名等,没有做必要的合法性校验
// 需要补充上这一部分. 校验部分也可以放在resolve,不过跟execution放一起也没有关系
RC ExecuteStage::do_select(const char *db, Query *sql, SessionEvent *session_event)
{
RC rc = RC::SUCCESS;
Session *session = session_event->get_client()->session;
Trx *trx = session->current_trx();
const Selects &selects = sql->sstr.selection;
// 把所有的表和只跟这张表关联的condition都拿出来,生成最底层的select 执行节点
std::vector<SelectExeNode *> select_nodes;
for (size_t i = 0; i < selects.relation_num; i++) {
const char *table_name = selects.relations[i];
SelectExeNode *select_node = new SelectExeNode;
rc = create_selection_executor(trx, selects, db, table_name, *select_node);
if (rc != RC::SUCCESS) {
delete select_node;
for (SelectExeNode *&tmp_node : select_nodes) {
delete tmp_node;
}
end_trx_if_need(session, trx, false);
return rc;
}
select_nodes.push_back(select_node);
}
if (select_nodes.empty()) {
LOG_ERROR("No table given");
end_trx_if_need(session, trx, false);
return RC::SQL_SYNTAX;
}
std::vector<TupleSet> tuple_sets;
for (SelectExeNode *&node : select_nodes) {
TupleSet tuple_set;
rc = node->execute(tuple_set);
if (rc != RC::SUCCESS) {
for (SelectExeNode *&tmp_node : select_nodes) {
delete tmp_node;
}
end_trx_if_need(session, trx, false);
return rc;
} else {
tuple_sets.push_back(std::move(tuple_set));
}
}
std::stringstream ss;
if (tuple_sets.size() > 1) {
// 本次查询了多张表,需要做join操作
} else {
// 当前只查询一张表,直接返回结果即可
tuple_sets.front().print(ss);
}
for (SelectExeNode *&tmp_node : select_nodes) {
delete tmp_node;
}
session_event->set_response(ss.str());
end_trx_if_need(session, trx, true);
return rc;
}
bool match_table(const Selects &selects, const char *table_name_in_condition, const char *table_name_to_match)
{
if (table_name_in_condition != nullptr) {
return 0 == strcmp(table_name_in_condition, table_name_to_match);
}
return selects.relation_num == 1;
}
static RC schema_add_field(Table *table, const char *field_name, TupleSchema &schema)
{
const FieldMeta *field_meta = table->table_meta().field(field_name);
if (nullptr == field_meta) {
LOG_WARN("No such field. %s.%s", table->name(), field_name);
return RC::SCHEMA_FIELD_MISSING;
}
schema.add_if_not_exists(field_meta->type(), table->name(), field_meta->name());
return RC::SUCCESS;
}
// 把所有的表和只跟这张表关联的condition都拿出来,生成最底层的select 执行节点
RC create_selection_executor(
Trx *trx, const Selects &selects, const char *db, const char *table_name, SelectExeNode &select_node)
{
// 列出跟这张表关联的Attr
TupleSchema schema;
Table *table = DefaultHandler::get_default().find_table(db, table_name);
if (nullptr == table) {
LOG_WARN("No such table [%s] in db [%s]", table_name, db);
return RC::SCHEMA_TABLE_NOT_EXIST;
}
for (int i = selects.attr_num - 1; i >= 0; i--) {
const RelAttr &attr = selects.attributes[i];
if (nullptr == attr.relation_name || 0 == strcmp(table_name, attr.relation_name)) {
if (0 == strcmp("*", attr.attribute_name)) {
// 列出这张表所有字段
TupleSchema::from_table(table, schema);
break; // 没有校验,给出* 之后,再写字段的错误
} else {
// 列出这张表相关字段
RC rc = schema_add_field(table, attr.attribute_name, schema);
if (rc != RC::SUCCESS) {
return rc;
}
}
}
}
// 找出仅与此表相关的过滤条件, 或者都是值的过滤条件
std::vector<DefaultConditionFilter *> condition_filters;
for (size_t i = 0; i < selects.condition_num; i++) {
const Condition &condition = selects.conditions[i];
if ((condition.left_is_attr == 0 && condition.right_is_attr == 0) || // 两边都是值
(condition.left_is_attr == 1 && condition.right_is_attr == 0 &&
match_table(selects, condition.left_attr.relation_name, table_name)) || // 左边是属性右边是值
(condition.left_is_attr == 0 && condition.right_is_attr == 1 &&
match_table(selects, condition.right_attr.relation_name, table_name)) || // 左边是值,右边是属性名
(condition.left_is_attr == 1 && condition.right_is_attr == 1 &&
match_table(selects, condition.left_attr.relation_name, table_name) &&
match_table(selects, condition.right_attr.relation_name, table_name)) // 左右都是属性名,并且表名都符合
) {
DefaultConditionFilter *condition_filter = new DefaultConditionFilter();
RC rc = condition_filter->init(*table, condition);
if (rc != RC::SUCCESS) {
delete condition_filter;
for (DefaultConditionFilter *&filter : condition_filters) {
delete filter;
}
return rc;
}
condition_filters.push_back(condition_filter);
}
}
return select_node.init(trx, table, std::move(schema), std::move(condition_filters));
}
#endif
RC
ExecuteStage
::
do_help
(
SQLStageEvent
*
sql_event
)
{
SessionEvent
*
session_event
=
sql_event
->
session_event
();
...
...
src/observer/sql/expr/expression.cpp
浏览文件 @
1cfc6080
/* Copyright (c) 2021 Xie Meiyi(xiemeiyi@hust.edu.cn) and OceanBase and/or its affiliates. All rights reserved.
miniob is licensed under Mulan PSL v2.
You can use this software according to the terms and conditions of the Mulan PSL v2.
You may obtain a copy of Mulan PSL v2 at:
http://license.coscl.org.cn/MulanPSL2
THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
See the Mulan PSL v2 for more details. */
//
// Created by Wangyunlai on 2022/07/05.
//
#include "sql/expr/tuple.h"
...
...
@@ -8,5 +22,5 @@ RC FieldExpr::get_value(const Tuple &tuple, TupleCell &cell) const
RC
ValueExpr
::
get_value
(
const
Tuple
&
tuple
,
TupleCell
&
cell
)
const
{
cell
=
tuple_cell
;
cell
=
tuple_cell
_
;
}
src/observer/sql/expr/expression.h
浏览文件 @
1cfc6080
/* Copyright (c) 2021 Xie Meiyi(xiemeiyi@hust.edu.cn) and OceanBase and/or its affiliates. All rights reserved.
miniob is licensed under Mulan PSL v2.
You can use this software according to the terms and conditions of the Mulan PSL v2.
You may obtain a copy of Mulan PSL v2 at:
http://license.coscl.org.cn/MulanPSL2
THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
See the Mulan PSL v2 for more details. */
//
// Created by Wangyunlai on 2022/07/05.
//
#pragma once
#include "storage/common/field.h"
...
...
@@ -5,6 +19,12 @@
class
Tuple
;
enum
class
ExprType
{
NONE
,
FIELD
,
VALUE
,
};
class
Expression
{
public:
...
...
@@ -12,6 +32,7 @@ public:
virtual
~
Expression
()
=
default
;
virtual
RC
get_value
(
const
Tuple
&
tuple
,
TupleCell
&
cell
)
const
=
0
;
virtual
ExprType
type
()
const
=
0
;
};
class
FieldExpr
:
public
Expression
...
...
@@ -23,6 +44,11 @@ public:
virtual
~
FieldExpr
()
=
default
;
ExprType
type
()
const
override
{
return
ExprType
::
FIELD
;
}
Field
&
field
()
{
return
field_
;
...
...
@@ -52,13 +78,21 @@ class ValueExpr : public Expression
{
public:
ValueExpr
()
=
default
;
ValueExpr
(
const
Value
&
value
)
:
tuple_cell
(
value
.
type
,
(
char
*
)
value
.
data
)
ValueExpr
(
const
Value
&
value
)
:
tuple_cell
_
(
value
.
type
,
(
char
*
)
value
.
data
)
{}
virtual
~
ValueExpr
()
=
default
;
RC
get_value
(
const
Tuple
&
tuple
,
TupleCell
&
cell
)
const
override
;
ExprType
type
()
const
override
{
return
ExprType
::
VALUE
;
}
void
get_tuple_cell
(
TupleCell
&
cell
)
const
{
cell
=
tuple_cell_
;
}
private:
TupleCell
tuple_cell
;
TupleCell
tuple_cell
_
;
};
src/observer/sql/expr/tuple.h
浏览文件 @
1cfc6080
...
...
@@ -9,7 +9,7 @@ MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
See the Mulan PSL v2 for more details. */
//
// Created by
Meiyi &
Wangyunlai on 2021/5/14.
// Created by Wangyunlai on 2021/5/14.
//
#pragma once
...
...
@@ -90,7 +90,7 @@ public:
this
->
record_
=
record
;
}
void
set_schema
(
Table
*
table
,
const
std
::
vector
<
FieldMeta
>
*
fields
)
void
set_schema
(
const
Table
*
table
,
const
std
::
vector
<
FieldMeta
>
*
fields
)
{
table_
=
table
;
this
->
speces_
.
reserve
(
fields
->
size
());
...
...
@@ -158,7 +158,7 @@ public:
}
private:
Record
*
record_
=
nullptr
;
Table
*
table_
=
nullptr
;
const
Table
*
table_
=
nullptr
;
std
::
vector
<
TupleCellSpec
*>
speces_
;
};
...
...
src/observer/sql/expr/tuple_cell.cpp
浏览文件 @
1cfc6080
/* Copyright (c) 2021 Xie Meiyi(xiemeiyi@hust.edu.cn) and OceanBase and/or its affiliates. All rights reserved.
miniob is licensed under Mulan PSL v2.
You can use this software according to the terms and conditions of the Mulan PSL v2.
You may obtain a copy of Mulan PSL v2 at:
http://license.coscl.org.cn/MulanPSL2
THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
See the Mulan PSL v2 for more details. */
//
// Created by WangYunlai on 2022/07/05.
//
#include "sql/expr/tuple_cell.h"
#include "storage/common/field.h"
#include "common/log/log.h"
...
...
src/observer/sql/expr/tuple_cell.h
浏览文件 @
1cfc6080
/* Copyright (c) 2021 Xie Meiyi(xiemeiyi@hust.edu.cn) and OceanBase and/or its affiliates. All rights reserved.
miniob is licensed under Mulan PSL v2.
You can use this software according to the terms and conditions of the Mulan PSL v2.
You may obtain a copy of Mulan PSL v2 at:
http://license.coscl.org.cn/MulanPSL2
THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
See the Mulan PSL v2 for more details. */
//
// Created by WangYunlai on 2022/6/7.
//
#pragma once
#include <iostream>
...
...
@@ -24,6 +38,15 @@ public:
int
compare
(
const
TupleCell
&
other
)
const
;
const
char
*
data
()
const
{
return
data_
;
}
AttrType
attr_type
()
const
{
return
attr_type_
;
}
private:
AttrType
attr_type_
=
UNDEFINED
;
char
*
data_
=
nullptr
;
// real data. no need to move to field_meta.offset
...
...
src/observer/sql/operator/index_scan_operator.cpp
0 → 100644
浏览文件 @
1cfc6080
#include "sql/operator/index_scan_operator.h"
#include "storage/index/index.h"
IndexScanOperator
::
IndexScanOperator
(
const
Table
*
table
,
Index
*
index
,
const
TupleCell
*
left_cell
,
bool
left_inclusive
,
const
TupleCell
*
right_cell
,
bool
right_inclusive
)
:
table_
(
table
),
index_
(
index
),
left_inclusive_
(
left_inclusive
),
right_inclusive_
(
right_inclusive
)
{
if
(
left_cell
)
{
left_cell_
=
*
left_cell
;
}
if
(
right_cell
)
{
right_cell_
=
*
right_cell
;
}
}
RC
IndexScanOperator
::
open
()
{
if
(
nullptr
==
table_
||
nullptr
==
index_
)
{
return
RC
::
INTERNAL
;
}
IndexScanner
*
index_scanner
=
index_
->
create_scanner
(
left_cell_
.
data
(),
left_inclusive_
,
right_cell_
.
data
(),
right_inclusive_
);
if
(
nullptr
==
index_scanner
)
{
LOG_WARN
(
"failed to create index scanner"
);
return
RC
::
INTERNAL
;
}
record_handler_
=
table_
->
record_handler
();
if
(
nullptr
==
record_handler_
)
{
LOG_WARN
(
"invalid record handler"
);
index_scanner
->
destroy
();
return
RC
::
INTERNAL
;
}
index_scanner_
=
index_scanner
;
tuple_
.
set_schema
(
table_
,
table_
->
table_meta
().
field_metas
());
return
RC
::
SUCCESS
;
}
RC
IndexScanOperator
::
next
()
{
RID
rid
;
RC
rc
=
index_scanner_
->
next_entry
(
&
rid
);
if
(
rc
!=
RC
::
SUCCESS
)
{
return
rc
;
}
return
record_handler_
->
get_record
(
&
rid
,
&
current_record_
);
}
RC
IndexScanOperator
::
close
()
{
index_scanner_
->
destroy
();
index_scanner_
=
nullptr
;
return
RC
::
SUCCESS
;
}
Tuple
*
IndexScanOperator
::
current_tuple
()
{
tuple_
.
set_record
(
&
current_record_
);
return
&
tuple_
;
}
src/observer/sql/operator/index_scan_operator.h
0 → 100644
浏览文件 @
1cfc6080
#pragma once
#include "sql/operator/operator.h"
#include "sql/expr/tuple.h"
class
IndexScanOperator
:
public
Operator
{
public:
IndexScanOperator
(
const
Table
*
table
,
Index
*
index
,
const
TupleCell
*
left_cell
,
bool
left_inclusive
,
const
TupleCell
*
right_cell
,
bool
right_inclusive
);
virtual
~
IndexScanOperator
()
=
default
;
RC
open
()
override
;
RC
next
()
override
;
RC
close
()
override
;
Tuple
*
current_tuple
()
override
;
private:
const
Table
*
table_
=
nullptr
;
Index
*
index_
=
nullptr
;
IndexScanner
*
index_scanner_
=
nullptr
;
RecordFileHandler
*
record_handler_
=
nullptr
;
Record
current_record_
;
RowTuple
tuple_
;
TupleCell
left_cell_
;
TupleCell
right_cell_
;
bool
left_inclusive_
;
bool
right_inclusive_
;
};
src/observer/storage/common/field.cpp
已删除
100644 → 0
浏览文件 @
cd892de3
src/observer/storage/common/field.h
浏览文件 @
1cfc6080
/* Copyright (c) 2021 Xie Meiyi(xiemeiyi@hust.edu.cn) and OceanBase and/or its affiliates. All rights reserved.
miniob is licensed under Mulan PSL v2.
You can use this software according to the terms and conditions of the Mulan PSL v2.
You may obtain a copy of Mulan PSL v2 at:
http://license.coscl.org.cn/MulanPSL2
THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
See the Mulan PSL v2 for more details. */
//
// Created by Wangyunlai on 2022/07/05.
//
#pragma once
#include "storage/common/table.h"
...
...
src/observer/storage/common/record.cpp
已删除
100644 → 0
浏览文件 @
cd892de3
#include "storage/common/record.h"
#include "storage/common/field.h"
#include "common/log/log.h"
#include "rc.h"
src/observer/storage/common/table.cpp
浏览文件 @
1cfc6080
...
...
@@ -737,6 +737,15 @@ Index *Table::find_index(const char *index_name) const
}
return
nullptr
;
}
Index
*
Table
::
find_index_by_field
(
const
char
*
field_name
)
const
{
const
TableMeta
&
table_meta
=
this
->
table_meta
();
const
IndexMeta
*
index_meta
=
table_meta
.
find_index_by_field
(
field_name
);
if
(
index_meta
!=
nullptr
)
{
return
this
->
find_index
(
index_meta
->
name
());
}
return
nullptr
;
}
IndexScanner
*
Table
::
find_index_for_scan
(
const
DefaultConditionFilter
&
filter
)
{
...
...
src/observer/storage/common/table.h
浏览文件 @
1cfc6080
...
...
@@ -64,6 +64,11 @@ public:
RC
get_record_scanner
(
RecordFileScanner
&
scanner
);
RecordFileHandler
*
record_handler
()
const
{
return
record_handler_
;
}
public:
const
char
*
name
()
const
;
...
...
@@ -98,8 +103,9 @@ private:
RC
init_record_handler
(
const
char
*
base_dir
);
RC
make_record
(
int
value_num
,
const
Value
*
values
,
char
*&
record_out
);
p
rivate
:
p
ublic
:
Index
*
find_index
(
const
char
*
index_name
)
const
;
Index
*
find_index_by_field
(
const
char
*
field_name
)
const
;
private:
std
::
string
base_dir_
;
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录