Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
CSDN 技术社区
skill_tree_oceanbase
提交
56030704
S
skill_tree_oceanbase
项目概览
CSDN 技术社区
/
skill_tree_oceanbase
通知
7
Star
4
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
DevOps
流水线
流水线任务
计划
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
S
skill_tree_oceanbase
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
DevOps
DevOps
流水线
流水线任务
计划
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
流水线任务
提交
Issue看板
提交
56030704
编写于
11月 07, 2021
作者:
M
Mars Liu
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
reflections for tree generator
上级
1de9c636
变更
3
隐藏空白更改
内联
并排
Showing
3 changed file
with
258 addition
and
189 deletion
+258
-189
data/tree.json
data/tree.json
+53
-53
main.py
main.py
+3
-2
src/tree.py
src/tree.py
+202
-134
未找到文件。
data/tree.json
浏览文件 @
56030704
...
...
@@ -251,55 +251,6 @@
]
}
},
{
"事务保存点"
:
{
"node_id"
:
"oceanbase-15dec616c0c14cdda056b3bacc20704f"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"事务超时"
:
{
"node_id"
:
"oceanbase-9e903dac84eb406d9bab6d66ce1a393e"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"创建和管理表"
:
{
"node_id"
:
"oceanbase-b1d157ec8b47413a8d6e3e2f08c87c19"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"创建和管理分区表"
:
{
"node_id"
:
"oceanbase-2cf15c8bbdca4f5a9b8bf760e83dc4cb"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"创建和管理表组"
:
{
"node_id"
:
"oceanbase-e927b436adc242fe8c2c46b60cccaf77"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"创建和管理视图"
:
{
"node_id"
:
"oceanbase-66b5429f4acd4e5489a1ae637fbcdc5a"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"迁移和同步"
:
{
"node_id"
:
"oceanbase-cb33c74506814a19b63a388042135240"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"开发者指南"
:
{
"node_id"
:
"oceanbase-2b5a186477dd4118a107392c965d7023"
,
...
...
@@ -412,6 +363,55 @@
"keywords"
:
[],
"children"
:
[]
}
},
{
"事务保存点"
:
{
"node_id"
:
"oceanbase-15dec616c0c14cdda056b3bacc20704f"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"事务超时"
:
{
"node_id"
:
"oceanbase-9e903dac84eb406d9bab6d66ce1a393e"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"创建和管理表"
:
{
"node_id"
:
"oceanbase-b1d157ec8b47413a8d6e3e2f08c87c19"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"创建和管理分区表"
:
{
"node_id"
:
"oceanbase-2cf15c8bbdca4f5a9b8bf760e83dc4cb"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"创建和管理表组"
:
{
"node_id"
:
"oceanbase-e927b436adc242fe8c2c46b60cccaf77"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"创建和管理视图"
:
{
"node_id"
:
"oceanbase-66b5429f4acd4e5489a1ae637fbcdc5a"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"迁移和同步"
:
{
"node_id"
:
"oceanbase-cb33c74506814a19b63a388042135240"
,
"keywords"
:
[],
"children"
:
[]
}
}
]
}
...
...
@@ -527,15 +527,15 @@
}
},
{
"
SQL请求执行顺序
"
:
{
"node_id"
:
"oceanbase-
9d1b0edd283647ec8e3d40b45731a6d
7"
,
"
函数
"
:
{
"node_id"
:
"oceanbase-
5c2cb3c282b7494bbf4b5f7e1fcb745
7"
,
"keywords"
:
[],
"children"
:
[]
}
},
{
"
函数
"
:
{
"node_id"
:
"oceanbase-
5c2cb3c282b7494bbf4b5f7e1fcb745
7"
,
"
SQL请求执行顺序
"
:
{
"node_id"
:
"oceanbase-
9d1b0edd283647ec8e3d40b45731a6d
7"
,
"keywords"
:
[],
"children"
:
[]
}
...
...
main.py
浏览文件 @
56030704
from
src.tree
import
gen_tree
from
src.tree
import
TreeWalker
if
__name__
==
'__main__'
:
gen_tree
(
'data'
)
walker
=
TreeWalker
(
"data"
,
"oceanbase"
,
"OceanBase"
)
walker
.
walk
()
src/tree.py
浏览文件 @
56030704
...
...
@@ -38,6 +38,7 @@ def ensure_config(path):
else
:
return
load_json
(
config_path
)
def
parse_no_name
(
d
):
p
=
r
'(\d+)\.(.*)'
m
=
re
.
search
(
p
,
d
)
...
...
@@ -65,142 +66,209 @@ def check_export(base, cfg):
return
flag
def
gen_tree
(
data_path
):
root
=
{}
def
gen_node_id
():
# return ''.join(str(uuid.uuid5(uuid.NAMESPACE_URL, 'skill_tree')).split('-'))
return
"oceanbase-"
+
uuid
.
uuid4
().
hex
def
list_dir
(
p
):
v
=
os
.
listdir
(
p
)
v
.
sort
()
for
no_name
in
v
:
no_dir
=
os
.
path
.
join
(
p
,
no_name
)
if
os
.
path
.
isdir
(
no_dir
):
yield
no_dir
,
no_name
def
ensure_id_helper
(
node
):
flag
=
False
if
(
node
.
get
(
'node_id'
)
is
None
)
or
node
.
get
(
'node_id'
)
in
id_set
:
node
[
'node_id'
]
=
gen_node_id
()
flag
=
True
id_set
.
add
(
node
[
'node_id'
])
if
'children'
in
node
:
for
c
in
node
[
"children"
]:
flag
=
flag
or
ensure_id_helper
(
list
(
c
.
values
())[
0
])
return
flag
def
ensure_node_id
(
cfg
):
return
ensure_id_helper
(
cfg
)
def
ensure_title_helper
(
node
,
cfg_path
,
title
=
""
):
flag
=
False
def
gen_node_id
():
return
"oceanbase-"
+
uuid
.
uuid4
().
hex
if
node
.
get
(
'title'
)
is
None
:
if
cfg_path
:
node
[
'title'
]
=
re
.
sub
(
"^[0-9]{1,3}\."
,
""
,
os
.
path
.
split
(
os
.
path
.
dirname
(
cfg_path
))[
-
1
])
else
:
node
[
'title'
]
=
title
flag
=
True
if
'children'
in
node
:
for
c
in
node
[
"children"
]:
flag
=
flag
or
ensure_title_helper
(
list
(
c
.
values
())[
0
],
None
,
list
(
c
.
keys
())[
0
])
return
flag
class
TreeWalker
:
def
__init__
(
self
,
root
,
tree_name
,
title
=
None
):
self
.
name
=
tree_name
self
.
root
=
root
self
.
title
=
tree_name
if
title
is
None
else
title
self
.
tree
=
{}
def
ensure_title
(
cfg
,
cfg_path
):
return
ensure_title_helper
(
cfg
,
cfg_path
)
def
walk
(
self
):
root
=
self
.
load_root
()
root_node
=
{
"node_id"
:
root
[
"node_id"
],
"keywords"
:
root
[
"keywords"
],
"children"
:
[]
}
self
.
tree
[
root
[
"tree_name"
]]
=
root_node
self
.
load_levels
(
root_node
)
self
.
load_chapters
(
self
.
root
,
root_node
)
for
index
,
level
in
enumerate
(
root_node
[
"children"
]):
level_title
=
list
(
level
.
keys
())[
0
]
level_node
=
list
(
level
.
values
())[
0
]
level_path
=
os
.
path
.
join
(
self
.
root
,
f
"
{
index
+
1
}
.
{
level_title
}
"
)
self
.
load_chapters
(
level_path
,
level_node
)
for
index
,
chapter
in
enumerate
(
level_node
[
"children"
]):
chapter_title
=
list
(
chapter
.
keys
())[
0
]
chapter_node
=
list
(
chapter
.
values
())[
0
]
chapter_path
=
os
.
path
.
join
(
level_path
,
f
"
{
index
+
1
}
.
{
chapter_title
}
"
)
self
.
load_sections
(
chapter_path
,
chapter_node
)
for
index
,
section_node
in
enumerate
(
chapter_node
[
"children"
]):
section_title
=
list
(
section_node
.
keys
())[
0
]
full_path
=
os
.
path
.
join
(
chapter_path
,
f
"
{
index
}
.
{
section_title
}
"
)
if
os
.
path
.
isdir
(
full_path
):
self
.
ensure_exercises
(
full_path
)
tree_path
=
os
.
path
.
join
(
self
.
root
,
"tree.json"
)
dump_json
(
tree_path
,
self
.
tree
,
exist_ok
=
True
,
override
=
True
)
return
self
.
tree
def
load_levels
(
self
,
root_node
):
levels
=
[]
for
level
in
os
.
listdir
(
self
.
root
):
if
not
os
.
path
.
isdir
(
level
):
continue
level_path
=
os
.
path
.
join
(
self
.
root
,
level
)
num
,
config
=
self
.
load_level_node
(
level_path
)
levels
.
append
((
num
,
config
))
levels
.
sort
(
key
=
lambda
item
:
item
[
0
])
root_node
[
"children"
]
=
[
item
[
1
]
for
item
in
levels
]
return
root_node
def
load_level_node
(
self
,
level_path
):
config
=
self
.
ensure_level_config
(
level_path
)
num
,
name
=
self
.
extract_node_env
(
level_path
)
result
=
{
name
:
{
"node_id"
:
config
[
"node_id"
],
"keywords"
:
config
[
"keywords"
],
"children"
:
[],
}
}
def
make_node
(
name
,
node_id
,
keywords
,
children
=
None
):
node
=
{}
node_children
=
children
or
[]
node
[
name
]
=
{
'node_id'
:
node_id
,
'keywords'
:
keywords
,
'children'
:
node_children
return
num
,
result
def
load_chapters
(
self
,
base
,
level_node
):
chapters
=
[]
for
name
in
os
.
listdir
(
base
):
full_name
=
os
.
path
.
join
(
base
,
name
)
if
os
.
path
.
isdir
(
full_name
):
num
,
chapter
=
self
.
load_chapter_node
(
full_name
)
chapters
.
append
((
num
,
chapter
))
chapters
.
sort
(
key
=
lambda
item
:
item
[
0
])
level_node
[
"children"
]
=
[
item
[
1
]
for
item
in
chapters
]
return
level_node
def
load_sections
(
self
,
base
,
chapter_node
):
sections
=
[]
for
name
in
os
.
listdir
(
base
):
full_name
=
os
.
path
.
join
(
base
,
name
)
if
os
.
path
.
isdir
(
full_name
):
num
,
section
=
self
.
load_section_node
(
full_name
)
sections
.
append
((
num
,
section
))
sections
.
sort
(
key
=
lambda
item
:
item
[
0
])
chapter_node
[
"children"
]
=
[
item
[
1
]
for
item
in
sections
]
return
chapter_node
def
ensure_chapters
(
self
):
for
subdir
in
os
.
listdir
(
self
.
root
):
self
.
ensure_level_config
(
subdir
)
def
load_root
(
self
):
config_path
=
os
.
path
.
join
(
self
.
root
,
"config.json"
)
if
not
os
.
path
.
exists
(
config_path
):
config
=
{
"tree_name"
:
self
.
name
,
"keywords"
:
[],
"node_id"
:
self
.
gen_node_id
(),
}
dump_json
(
config_path
,
config
,
exist_ok
=
True
,
override
=
True
)
else
:
config
=
load_json
(
config_path
)
flag
,
result
=
self
.
ensure_node_id
(
config
)
if
flag
:
dump_json
(
config_path
,
result
,
exist_ok
=
True
,
override
=
True
)
return
config
def
ensure_level_config
(
self
,
path
):
config_path
=
os
.
path
.
join
(
path
,
"config.json"
)
if
not
os
.
path
.
exists
(
config_path
):
config
=
{
"node_id"
:
self
.
gen_node_id
()
}
dump_json
(
path
,
config
,
exist_ok
=
True
,
override
=
True
)
else
:
config
=
load_json
(
config_path
)
flag
,
result
=
self
.
ensure_node_id
(
config
)
if
flag
:
dump_json
(
path
,
config
,
exist_ok
=
True
,
override
=
True
)
return
config
def
ensure_chapter_config
(
self
,
path
):
config_path
=
os
.
path
.
join
(
path
,
"config.json"
)
if
not
os
.
path
.
exists
(
config_path
):
config
=
{
"node_id"
:
self
.
gen_node_id
(),
"keywords"
:
[]
}
dump_json
(
path
,
config
,
exist_ok
=
True
,
override
=
True
)
else
:
config
=
load_json
(
config_path
)
flag
,
result
=
self
.
ensure_node_id
(
config
)
if
flag
:
dump_json
(
path
,
config
,
exist_ok
=
True
,
override
=
True
)
return
config
def
ensure_section_config
(
self
,
path
):
config_path
=
os
.
path
.
join
(
path
,
"config.json"
)
if
not
os
.
path
.
exists
(
config_path
):
config
=
{
"node_id"
:
self
.
gen_node_id
(),
"keywords"
:
[],
}
dump_json
(
config_path
,
config
,
exist_ok
=
True
,
override
=
True
)
else
:
config
=
load_json
(
config_path
)
flag
,
result
=
self
.
ensure_node_id
(
config
)
if
flag
:
dump_json
(
config_path
,
config
,
exist_ok
=
True
,
override
=
True
)
return
config
def
ensure_node_id
(
self
,
config
):
if
"node_id"
not
in
config
:
config
[
"node_id"
]
=
self
.
gen_node_id
()
return
True
,
config
else
:
return
False
,
config
def
gen_node_id
(
self
):
return
f
"
{
self
.
name
}
-
{
uuid
.
uuid4
().
hex
}
"
def
extract_node_env
(
self
,
path
):
_
,
dir
=
os
.
path
.
split
(
path
)
number
,
title
=
dir
.
split
(
"."
,
1
)
return
int
(
number
),
title
def
load_chapter_node
(
self
,
full_name
):
config
=
self
.
ensure_chapter_config
(
full_name
)
num
,
name
=
self
.
extract_node_env
(
full_name
)
result
=
{
name
:
{
"node_id"
:
config
[
"node_id"
],
"keywords"
:
config
[
"keywords"
],
"children"
:
[],
}
}
return
node
,
node_children
# 根节点
cfg
=
ensure_config
(
data_path
)
cfg_path
=
os
.
path
.
join
(
data_path
,
'config.json'
)
if
ensure_node_id
(
cfg
):
dump_json
(
cfg_path
,
cfg
,
exist_ok
=
True
,
override
=
True
)
if
ensure_title
(
cfg
,
cfg_path
):
cfg
[
"title"
]
=
"C"
dump_json
(
cfg_path
,
cfg
,
exist_ok
=
True
,
override
=
True
)
tree_node
=
{
"node_id"
:
cfg
[
'node_id'
],
"keywords"
:
cfg
[
'keywords'
],
"children"
:
[]
}
root
[
cfg
[
'tree_name'
]]
=
tree_node
# 难度节点
for
level_no_dir
,
level_no_name
in
list_dir
(
data_path
):
print
(
level_no_dir
)
no
,
level_name
=
parse_no_name
(
level_no_name
)
level_path
=
os
.
path
.
join
(
level_no_dir
,
'config.json'
)
level_cfg
=
ensure_config
(
level_no_dir
)
if
ensure_node_id
(
level_cfg
)
or
check_export
(
level_no_dir
,
level_cfg
):
dump_json
(
level_path
,
level_cfg
,
exist_ok
=
True
,
override
=
True
)
if
ensure_title
(
level_cfg
,
level_path
):
dump_json
(
level_path
,
level_cfg
,
exist_ok
=
True
,
override
=
True
)
level_node
,
level_node_children
=
make_node
(
level_name
,
level_cfg
[
'node_id'
],
level_cfg
[
'keywords'
])
tree_node
[
'children'
].
append
(
level_node
)
# 章节点
for
chapter_no_dir
,
chapter_no_name
in
list_dir
(
level_no_dir
):
no
,
chapter_name
=
parse_no_name
(
chapter_no_name
)
chapter_path
=
os
.
path
.
join
(
chapter_no_dir
,
'config.json'
)
chapter_cfg
=
ensure_config
(
chapter_no_dir
)
if
ensure_node_id
(
chapter_cfg
)
or
check_export
(
chapter_no_dir
,
chapter_cfg
):
dump_json
(
chapter_path
,
chapter_cfg
,
exist_ok
=
True
,
override
=
True
)
if
ensure_title
(
chapter_cfg
,
chapter_path
):
dump_json
(
chapter_path
,
chapter_cfg
,
exist_ok
=
True
,
override
=
True
)
chapter_node
,
chapter_node_children
=
make_node
(
chapter_name
,
chapter_cfg
[
'node_id'
],
chapter_cfg
[
'keywords'
])
level_node_children
.
append
(
chapter_node
)
# 知识点
for
section_no_dir
,
section_no_name
in
list_dir
(
chapter_no_dir
):
no
,
section_name
=
parse_no_name
(
section_no_name
)
sec_path
=
os
.
path
.
join
(
section_no_dir
,
'config.json'
)
sec_cfg
=
ensure_config
(
section_no_dir
)
flag
=
ensure_node_id
(
sec_cfg
)
or
check_export
(
section_no_dir
,
sec_cfg
)
section_node
,
section_node_children
=
make_node
(
section_name
,
sec_cfg
[
'node_id'
],
sec_cfg
[
'keywords'
],
sec_cfg
.
get
(
'children'
,
[]))
chapter_node_children
.
append
(
section_node
)
# 确保习题分配了习题ID
for
export
in
sec_cfg
.
get
(
"export"
,
[]):
ecfg_path
=
os
.
path
.
join
(
section_no_dir
,
export
)
ecfg
=
load_json
(
ecfg_path
)
if
(
ecfg
.
get
(
'exercise_id'
)
is
None
)
or
(
ecfg
.
get
(
'exercise_id'
)
in
id_set
):
ecfg
[
'exercise_id'
]
=
uuid
.
uuid4
().
hex
dump_json
(
ecfg_path
,
ecfg
,
exist_ok
=
True
,
override
=
True
)
id_set
.
add
(
ecfg
[
'exercise_id'
])
if
flag
:
dump_json
(
sec_path
,
sec_cfg
,
exist_ok
=
True
,
override
=
True
)
if
ensure_title
(
sec_cfg
,
sec_path
):
dump_json
(
sec_path
,
sec_cfg
,
exist_ok
=
True
,
override
=
True
)
# 保存技能树骨架
tree_path
=
os
.
path
.
join
(
data_path
,
'tree.json'
)
dump_json
(
tree_path
,
root
,
exist_ok
=
True
,
override
=
True
)
return
num
,
result
def
load_section_node
(
self
,
full_name
):
config
=
self
.
ensure_section_config
(
full_name
)
num
,
name
=
self
.
extract_node_env
(
full_name
)
result
=
{
name
:
{
"node_id"
:
config
[
"node_id"
],
"keywords"
:
config
[
"keywords"
],
"children"
:
config
.
get
(
"children"
,
[])
}
}
# if "children" in config:
# result["children"] = config["children"]
return
num
,
result
def
ensure_exercises
(
self
,
section_path
):
config
=
self
.
ensure_section_config
(
section_path
)
for
e
in
config
.
get
(
"export"
,
[]):
full_name
=
os
.
path
.
join
(
section_path
,
e
)
exercise
=
load_json
(
full_name
)
if
"exercise_id"
not
in
exercise
:
exercise
[
"exercise_id"
]
=
uuid
.
uuid4
().
hex
dump_json
(
full_name
,
exercise
)
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录