Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
2dot5
ClickHouse
提交
fdd34069
C
ClickHouse
项目概览
2dot5
/
ClickHouse
通知
3
Star
0
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
DevOps
流水线
流水线任务
计划
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
C
ClickHouse
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
DevOps
DevOps
流水线
流水线任务
计划
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
流水线任务
提交
Issue看板
体验新版 GitCode,发现更多精彩内容 >>
提交
fdd34069
编写于
9月 26, 2014
作者:
A
Alexey Milovidov
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
dbms: probably fixed error in cyclic ALTERs [#METR-12509].
上级
ba0191d0
变更
1
隐藏空白更改
内联
并排
Showing
1 changed file
with
50 addition
and
24 deletion
+50
-24
dbms/src/Storages/StorageReplicatedMergeTree.cpp
dbms/src/Storages/StorageReplicatedMergeTree.cpp
+50
-24
未找到文件。
dbms/src/Storages/StorageReplicatedMergeTree.cpp
浏览文件 @
fdd34069
...
...
@@ -1530,59 +1530,76 @@ void StorageReplicatedMergeTree::alterThread()
{
try
{
/** Имеем описание столбцов в ZooKeeper, общее для всех реплик (Пример: /clickhouse/tables/02-06/visits/columns),
* а также описание столбцов в локальном файле с метаданными (data.getColumnsList()).
*
* Если эти описания отличаются - нужно сделать ALTER.
*
* Если запомненная версия ноды (columns_version) отличается от версии в ZK,
* то описание столбцов в ZK не обязательно отличается от локального
* - такое может быть при цикле из ALTER-ов, который в целом, ничего не меняет.
* В этом случае, надо обновить запомненный номер версии,
* а также всё-равно проверить структуру кусков, и, при необходимости, сделать ALTER.
*
* Запомненный номер версии нужно обновить после обновления метаданных, под блокировкой.
* Этот номер версии проверяется на соответствие актуальному при INSERT-е.
* То есть, так добиваемся, чтобы вставлялись блоки с правильной структурой.
*
* При старте сервера, мог быть не завершён предыдущий ALTER.
* Поэтому, в первый раз, независимо от изменений, проверяем структуру всех part-ов,
* (Пример: /clickhouse/tables/02-06/visits/replicas/example02-06-1.yandex.ru/parts/20140806_20140831_131664_134988_3296/columns)
* и делаем ALTER, если необходимо.
*
* TODO: Слишком сложно, всё переделать.
*/
zkutil
::
Stat
stat
;
String
columns_str
=
zookeeper
->
get
(
zookeeper_path
+
"/columns"
,
&
stat
,
alter_thread_event
);
NamesAndTypesList
columns
=
NamesAndTypesList
::
parse
(
columns_str
,
context
.
getDataTypeFactory
());
bool
changed
=
false
;
/// Проверим, что описание столбцов изменилось.
/// Чтобы не останавливать лишний раз все запросы в таблицу, проверим сначала под локом на чтение.
{
auto
table_lock
=
lockStructure
(
false
);
if
(
columns
!=
data
.
getColumnsList
())
changed
=
true
;
}
bool
changed_version
=
(
stat
.
version
!=
columns_version
);
MergeTreeData
::
DataParts
parts
;
/// Если описание столбцов изменилось, обновим структуру таблицы локально.
if
(
changed
)
if
(
changed
_version
)
{
auto
table_lock
=
lockStructureForAlter
();
if
(
columns
!=
data
.
getColumnsList
())
{
LOG_INFO
(
log
,
"Columns list changed in ZooKeeper. Applying changes locally."
);
InterpreterAlterQuery
::
updateMetadata
(
database_name
,
table_name
,
columns
,
context
);
data
.
setColumnsList
(
columns
);
if
(
unreplicated_data
)
unreplicated_data
->
setColumnsList
(
columns
);
columns_version
=
stat
.
version
;
LOG_INFO
(
log
,
"Applied changes to table."
);
/// Нужно получить список кусков под блокировкой таблицы, чтобы избежать race condition с мерджем.
parts
=
data
.
getDataParts
();
}
else
{
changed
=
false
;
columns_version
=
stat
.
version
;
LOG_INFO
(
log
,
"Columns version changed in ZooKeeper, but data wasn't changed. It's like cyclic ALTERs."
);
}
/// Нужно получить список кусков под блокировкой таблицы, чтобы избежать race condition с мерджем.
parts
=
data
.
getDataParts
();
columns_version
=
stat
.
version
;
}
/// Обновим куски.
if
(
changed
||
force_recheck_parts
)
if
(
changed
_version
||
force_recheck_parts
)
{
if
(
changed
)
auto
table_lock
=
lockStructure
(
false
);
if
(
changed_version
)
LOG_INFO
(
log
,
"ALTER-ing parts"
);
int
changed_parts
=
0
;
if
(
!
changed
)
if
(
!
changed
_version
)
parts
=
data
.
getDataParts
();
auto
table_lock
=
lockStructure
(
false
);
for
(
const
MergeTreeData
::
DataPartPtr
&
part
:
parts
)
{
/// Обновим кусок и запишем результат во временные файлы.
...
...
@@ -1623,10 +1640,17 @@ void StorageReplicatedMergeTree::alterThread()
}
}
/// Список столбцов для конкретной реплики.
zookeeper
->
set
(
replica_path
+
"/columns"
,
columns
.
toString
());
if
(
changed
||
changed_parts
!=
0
)
LOG_INFO
(
log
,
"ALTER-ed "
<<
changed_parts
<<
" parts"
);
if
(
changed_version
)
{
if
(
changed_parts
!=
0
)
LOG_INFO
(
log
,
"ALTER-ed "
<<
changed_parts
<<
" parts"
);
else
LOG_INFO
(
log
,
"No parts ALTER-ed"
);
}
force_recheck_parts
=
false
;
}
...
...
@@ -1642,7 +1666,7 @@ void StorageReplicatedMergeTree::alterThread()
}
}
LOG_DEBUG
(
log
,
"
a
lter thread finished"
);
LOG_DEBUG
(
log
,
"
A
lter thread finished"
);
}
void
StorageReplicatedMergeTree
::
removePartAndEnqueueFetch
(
const
String
&
part_name
)
...
...
@@ -2284,6 +2308,7 @@ void StorageReplicatedMergeTree::alter(const AlterCommands & params,
/// Подпишемся на изменения столбцов, чтобы перестать ждать, если кто-то еще сделает ALTER.
if
(
!
zookeeper
->
exists
(
zookeeper_path
+
"/columns"
,
&
stat
,
alter_query_event
))
throw
Exception
(
zookeeper_path
+
"/columns doesn't exist"
,
ErrorCodes
::
NOT_FOUND_NODE
);
if
(
stat
.
version
!=
new_columns_version
)
{
LOG_WARNING
(
log
,
zookeeper_path
+
"/columns changed before this ALTER finished; "
...
...
@@ -2314,6 +2339,7 @@ void StorageReplicatedMergeTree::alter(const AlterCommands & params,
if
(
!
zookeeper
->
exists
(
zookeeper_path
+
"/columns"
,
&
stat
))
throw
Exception
(
zookeeper_path
+
"/columns doesn't exist"
,
ErrorCodes
::
NOT_FOUND_NODE
);
if
(
stat
.
version
!=
new_columns_version
)
{
LOG_WARNING
(
log
,
zookeeper_path
+
"/columns changed before ALTER finished; "
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录