未验证 提交 7e6a6d2a 编写于 作者: X Xu Wenhao 提交者: GitHub

build: develop by vscode in docker container (#100) (#125)

优化Docker开发环境:
- docker镜像使用zsh;
- 自动初始化vscode开发调试配置环境;
- 容器中开启ssh服务,方便远程开发;
- 可以在构建镜像时clone自己的代码;
- 提供了丰富详细的使用文档。
上级 e279dc7a
...@@ -15,6 +15,7 @@ miniob 设计的目标是让同学们快速了解数据库并深入学习数据 ...@@ -15,6 +15,7 @@ miniob 设计的目标是让同学们快速了解数据库并深入学习数据
1. [本地配置gcc环境](docs/how_to_build.md) 1. [本地配置gcc环境](docs/how_to_build.md)
2. [使用Docker开发](docs/how-to-dev-using-docker.md) 2. [使用Docker开发](docs/how-to-dev-using-docker.md)
3. [在Windows上使用Docker](docs/how_to_dev_miniob_by_docker_on_windows.md) 3. [在Windows上使用Docker](docs/how_to_dev_miniob_by_docker_on_windows.md)
4. [使用 Docker 和 VSCode 远程开发 Miniob (推荐)](docs/how_to_dev_in_docker_container_by_vscode.md)
## 词法、语法解析 ## 词法、语法解析
请参考 [miniob 词法语法解析开发与测试](docs/miniob-sql-parser.md) 请参考 [miniob 词法语法解析开发与测试](docs/miniob-sql-parser.md)
......
...@@ -5,7 +5,7 @@ ...@@ -5,7 +5,7 @@
FROM openanolis/anolisos:8.6 FROM openanolis/anolisos:8.6
ARG HOME_DIR=/root ARG HOME_DIR=/root
ARG GIT_SOURCE=github ARG DOCKER_CONFIG_DIR=${HOME_DIR}/docker
ENV LANG=en_US.UTF-8 ENV LANG=en_US.UTF-8
...@@ -16,28 +16,23 @@ RUN dnf install -y make cmake git wget which flex gdb gcc gcc-c++ diffutils read ...@@ -16,28 +16,23 @@ RUN dnf install -y make cmake git wget which flex gdb gcc gcc-c++ diffutils read
# rockylinux:8 # rockylinux:8
# RUN dnf --enablerepo=powertools install -y texinfo # RUN dnf --enablerepo=powertools install -y texinfo
# prepare env
WORKDIR ${HOME_DIR}
RUN echo alias ls=\"ls --color=auto\" >> .bashrc
RUN echo "export LD_LIBRARY_PATH=/usr/local/lib64:\$LD_LIBRARY_PATH" >> .bashrc
# clone deps and compile deps # clone deps and compile deps
RUN mkdir -p ${HOME_DIR}/deps # RUN mkdir -p ${HOME_DIR}/deps
WORKDIR ${HOME_DIR}/deps WORKDIR ${HOME_DIR}/deps
RUN git clone https://github.com/libevent/libevent -b release-2.1.12-stable \ RUN git clone https://gitee.com/mirrors/libevent.git -b release-2.1.12-stable \
&& mkdir -p ${HOME_DIR}/deps/libevent/build \ && mkdir -p ${HOME_DIR}/deps/libevent/build \
&& cmake -DEVENT__DISABLE_OPENSSL=ON -B ${HOME_DIR}/deps/libevent/build ${HOME_DIR}/deps/libevent \ && cmake -DEVENT__DISABLE_OPENSSL=ON -B ${HOME_DIR}/deps/libevent/build ${HOME_DIR}/deps/libevent \
&& make -C ${HOME_DIR}/deps/libevent/build -j install \ && make -C ${HOME_DIR}/deps/libevent/build -j install \
&& rm -rf ${HOME_DIR}/deps/* && rm -rf ${HOME_DIR}/deps/*
RUN git clone https://github.com/open-source-parsers/jsoncpp.git \ RUN git clone https://gitee.com/mirrors/jsoncpp.git \
&& mkdir -p ${HOME_DIR}/deps/jsoncpp/build \ && mkdir -p ${HOME_DIR}/deps/jsoncpp/build \
&& cmake -DJSONCPP_WITH_TESTS=OFF -DJSONCPP_WITH_POST_BUILD_UNITTEST=OFF -B ${HOME_DIR}/deps/jsoncpp/build ${HOME_DIR}/deps/jsoncpp/ \ && cmake -DJSONCPP_WITH_TESTS=OFF -DJSONCPP_WITH_POST_BUILD_UNITTEST=OFF -B ${HOME_DIR}/deps/jsoncpp/build ${HOME_DIR}/deps/jsoncpp/ \
&& make -C ${HOME_DIR}/deps/jsoncpp/build -j install \ && make -C ${HOME_DIR}/deps/jsoncpp/build -j install \
&& rm -rf ${HOME_DIR}/deps/* && rm -rf ${HOME_DIR}/deps/*
RUN git clone https://github.com/google/googletest \ RUN git clone https://gitee.com/mirrors/googletest.git \
&& mkdir -p ${HOME_DIR}/deps/googletest/build \ && mkdir -p ${HOME_DIR}/deps/googletest/build \
&& cmake -B ${HOME_DIR}/deps/googletest/build ${HOME_DIR}/deps/googletest \ && cmake -B ${HOME_DIR}/deps/googletest/build ${HOME_DIR}/deps/googletest \
&& make -C ${HOME_DIR}/deps/googletest/build -j install \ && make -C ${HOME_DIR}/deps/googletest/build -j install \
...@@ -50,18 +45,32 @@ RUN wget http://ftp.gnu.org/gnu/bison/bison-3.7.tar.gz \ ...@@ -50,18 +45,32 @@ RUN wget http://ftp.gnu.org/gnu/bison/bison-3.7.tar.gz \
&& make -j install \ && make -j install \
&& rm -rf ${HOME_DIR}/deps/* && rm -rf ${HOME_DIR}/deps/*
# clone miniob code # install openssh
RUN mkdir -p ${HOME_DIR}/source RUN yum -y install openssh openssh-clients openssh-server
WORKDIR ${HOME_DIR}/source
RUN git clone https://${GIT_SOURCE}.com/oceanbase/miniob # change root password
RUN echo "mkdir -p build && cd build && cmake .. -DDEBUG=ON -DCMAKE_C_COMPILER=`which gcc` -DCMAKE_CXX_COMPILER=`which g++` && make -j4" > ${HOME_DIR}/source/miniob/build.sh && chmod +x ${HOME_DIR}/source/miniob/build.sh RUN echo "root:root" | chpasswd
RUN mkdir -p ${HOME_DIR}/source/miniob/build RUN mkdir /var/run/sshd
WORKDIR ${HOME_DIR}/source/miniob/build
RUN cmake -B ${HOME_DIR}/source/miniob/build -DDEBUG=ON -DCMAKE_C_COMPILER=`which gcc` -DCMAKE_CXX_COMPILER=`which g++` ${HOME_DIR}/source/miniob/ \ # install zsh and on-my-zsh
&& make -j4 \ RUN yum install -y zsh \
&& rm -rf ${HOME_DIR}/source/miniob/build && git clone https://gitee.com/mirrors/ohmyzsh.git ~/.oh-my-zsh \
&& cp ~/.oh-my-zsh/templates/zshrc.zsh-template ~/.zshrc \
&& sed -i "s/robbyrussell/bira/" ~/.zshrc \
&& usermod --shell /bin/zsh root \
&& echo "export LD_LIBRARY_PATH=/usr/local/lib64:\$LD_LIBRARY_PATH" >> ~/.zshrc
RUN mkdir -p ${DOCKER_CONFIG_DIR}/bin
RUN mkdir -p ${DOCKER_CONFIG_DIR}/config
WORKDIR ${DOCKER_CONFIG_DIR}/bin
# copy vscode config files
COPY config/* ${DOCKER_CONFIG_DIR}/config/
WORKDIR ${HOME_DIR} # copy starter scripts
COPY bin/* ${DOCKER_CONFIG_DIR}/bin/
ENTRYPOINT tail -f /dev/null RUN chmod +x ${DOCKER_CONFIG_DIR}/bin/*
ENTRYPOINT ./starter.sh
\ No newline at end of file
#!/bin/bash
# generate ssh private key by env argument
echo "Try to clone repository..."
SSH_DIR=/root/.ssh
if [ "$PRIVATE_KEY" = "" ]; then
echo "ENV PRIVATE_KEY is not set."
else
if [ -d "${SSH_DIR}" ]; then
echo "folder ~/.ssh exists..."
else
echo "Generating SSH private rsa key..."
mkdir -p ${SSH_DIR}
echo "-----BEGIN OPENSSH PRIVATE KEY-----" >${SSH_DIR}/id_rsa
echo $PRIVATE_KEY | tr -s " " "\n" | sed "1,4d" | head -n -4 | xargs -L 1 >>${SSH_DIR}/id_rsa
echo "-----END OPENSSH PRIVATE KEY-----" >>${SSH_DIR}/id_rsa
chmod 700 ${SSH_DIR}
chmod 600 ${SSH_DIR}/id_rsa
# add SSH key fingerprint to known_hosts
echo $REPO_ADDR | awk -F'[@:]' '{print $2}' | xargs ssh-keyscan $1 >>${SSH_DIR}/known_hosts
echo "SSH private rsa key generated!"
fi
fi
# check if source code exists
REPO_DIR=/root/source/miniob
if [ -d "${REPO_DIR}" ]; then
cd ${REPO_DIR}
is_git_repo=$(git rev-parse --is-inside-work-tree 2>&1)
if [ "${is_git_repo}" = "true" ]; then
echo "The source code has been cloned!"
exit 0
else
echo "WARNING: /root/source/miniob exists but it is not a git repository."
exit 0
fi
fi
# clone code.
if [ "$REPO_ADDR" = "" ]; then
echo "ENV REPO_ADDR is not set."
exit 0
else
WORK_DIR=/root/source
mkdir -p ${WORK_DIR}
git clone $REPO_ADDR ${REPO_DIR}
fi
if [ ! -d "${REPO_DIR}/bin" ]; then
mkdir -p ${REPO_DIR}/build
fi
# vscode config
if [ ! -d "${REPO_DIR}/.vscode" ]; then
mkdir -p ${REPO_DIR}/.vscode
fi
if [ ! -f "${REPO_DIR}/.vscode/launch.json" ]; then
cp /root/docker/config/launch.json ${REPO_DIR}/.vscode/launch.json
fi
if [ ! -f "${REPO_DIR}/.vscode/tasks.json" ]; then
cp /root/docker/config/tasks.json ${REPO_DIR}/.vscode/tasks.json
fi
#!/bin/bash
HOST_KEY_DIR=/etc/ssh/ssh_host_rsa_key
if [ ! -f "${HOST_KEY_DIR}" ]; then
ssh-keygen -A
fi
/usr/sbin/sshd -D
echo sshd started!
#!/bin/bash
ls -lld $PWD/*starter-* | awk '{print $9;}' | xargs -L 1 bash -c
echo 'starter scripts run successfully!'
tail -f /dev/null
{
// 使用 IntelliSense 了解相关属性。
// 悬停以查看现有属性的描述。
// 欲了解更多信息,请访问: https://go.microsoft.com/fwlink/?linkid=830387
"version": "0.2.0",
"configurations": [
{
"name": "(gdb) 启动",
"type": "cppdbg",
"request": "launch",
"program": "${workspaceFolder}/build/bin/observer",
"args": ["-f", "${workspaceFolder}/etc/observer.ini"],
"stopAtEntry": false,
"cwd": "${fileDirname}",
"environment": [],
"externalConsole": false,
"MIMode": "gdb",
"setupCommands": [
{
"description": "为 gdb 启用整齐打印",
"text": "-enable-pretty-printing",
"ignoreFailures": true
},
{
"description": "将反汇编风格设置为 Intel",
"text": "-gdb-set disassembly-flavor intel",
"ignoreFailures": true
}
],
"preLaunchTask": "CMake Build"
}
]
}
\ No newline at end of file
{
"options": {
"cwd": "${workspaceFolder}/build"
},
"tasks": [
{
"label": "cmake",
"command": "cmake",
"args": [
"-DDEBUG=ON",
".."
]
},
{
"label": "make",
"command": "make",
"args": [
"-j",
"6"
]
},
{
"label": "CMake Build",
"dependsOn": [
"cmake",
"make"
]
}
],
"version": "2.0.0"
}
\ No newline at end of file
version: "3.3"
services:
miniob-dev:
build:
context: .
dockerfile: Dockerfile
image: miniob:v1.0
container_name: miniob
privileged: true
ports:
- "10000:22"
restart: unless-stopped
environment:
# set the env argument PRIVATE_KEY when using ssh clone
# REPO_ADDR format in a private repo: https://<username>:<password>@github.com/oceanbase/miniob.git, '@' in username must be escape to '%40'
# How to run:
# Linux shell:
# export PRIVATE_KEY=$(cat ~/.ssh/id_rsa) && docker-compose up -d --build
# Windows powershell:
# $env:PRIVATE_KEY=$(cat ~/.ssh/id_rsa) && docker-compose up -d --build
- REPO_ADDR=https://github.com/oceanbase/miniob.git
- PRIVATE_KEY=${PRIVATE_KEY}
\ No newline at end of file
...@@ -6,10 +6,11 @@ MiniOB 依赖的第三方组件比较多,搭建开发环境比较繁琐,建 ...@@ -6,10 +6,11 @@ MiniOB 依赖的第三方组件比较多,搭建开发环境比较繁琐,建
如果对Docker还不太熟悉,可以先在网上大致了解一下。 如果对Docker还不太熟悉,可以先在网上大致了解一下。
我们提供了原始的Dockerfile,也有已经打包好的镜像,可以选择自己喜欢的方式。 我们提供了原始的Dockerfile,也有已经打包好的镜像,可以选择自己喜欢的方式。
自行构建参考[本文档](./how_to_dev_in_docker_container_by_vscode.md)
- 使用 Dockerfile 构建 - 使用 Dockerfile 构建
Dockerfile: https://github.com/oceanbase/miniob/blob/main/Dockerfile Dockerfile: https://github.com/oceanbase/miniob/blob/main/docker/Dockerfile
```bash ```bash
# build # build
...@@ -17,7 +18,7 @@ docker build -t miniob . ...@@ -17,7 +18,7 @@ docker build -t miniob .
docker run --privileged -d --name='miniob' miniob:latest docker run --privileged -d --name='miniob' miniob:latest
# 进入容器,开发调试 # 进入容器,开发调试
docker exec -it miniob bash docker exec -it miniob /usr/bin/zsh
``` ```
- 使用docker hub 镜像运行 - 使用docker hub 镜像运行
......
# 使用 Docker 和 VSCode 远程开发 Miniob
本文档阐述如何使用 Docker 构建一个具备 Miniob 开发环境的容器,并且通过 VSCode 的 Remote-SSH 插件 SSH 到容器中进行远程开发。Docker 和 VSCode 可以安装在不同机器上。
方法简单易行,仅需配置 2 个 环境变量(仓库 URL、SSH秘钥)及安装必要的 VSCode 开发插件即可。
- [使用 Docker 和 VSCode 远程开发 Miniob](#使用-docker-和-vscode-远程开发-miniob)
- [安装 Docker 和 Docker-compose](#安装-docker-和-docker-compose)
- [Windows](#windows)
- [Linux](#linux)
- [安装 VSCode](#安装-vscode)
- [配置 Dockerfile](#配置-dockerfile)
- [配置 Root 密码](#配置-root-密码)
- [配置 Docker Compose](#配置-docker-compose)
- [Fork 仓库](#fork-仓库)
- [Clone by SSH](#clone-by-ssh)
- [Clone by HTTPS](#clone-by-https)
- [私有仓库](#私有仓库)
- [公有仓库](#公有仓库)
- [配置映射端口](#配置映射端口)
- [启动容器](#启动容器)
- [Clone by SSH](#clone-by-ssh-1)
- [Linux Shell](#linux-shell)
- [Windows Powershell](#windows-powershell)
- [Clone by HTTPS](#clone-by-https-1)
- [管理容器](#管理容器)
- [使用 VSCode 远程开发](#使用-vscode-远程开发)
- [安装 Remote-SSH 插件](#安装-remote-ssh-插件)
- [连接 Docker 容器](#连接-docker-容器)
- [配置 SSH Host](#配置-ssh-host)
- [连接容器](#连接容器)
- [打开代码目录](#打开代码目录)
- [安装开发插件](#安装开发插件)
- [进行 Debug](#进行-debug)
- [启动 Server](#启动-server)
- [启动 Client](#启动-client)
- [执行 SQL](#执行-sql)
## 安装 Docker 和 Docker-compose
### Windows
参考本仓库文档 [在 Windows 上使用 Docker](how_to_dev_miniob_by_docker_on_windows.md) 安装 Docker Desktop 即可。只需执行 **安装 Docker** 这一小节。
### Linux
如果你选择在远程建议参考官方文档安装 [Docker Engine](https://docs.docker.com/engine/install/)[Docker Compose](https://docs.docker.com/compose/install/linux/).
## 安装 VSCode
前往官网下载 [Visual Studio Code](https://code.visualstudio.com/),正常安装即可。
## 配置 Dockerfile
文件在 docker/Dockerfile.
### 配置 Root 密码
默认密码 root,可在 64 行自行修改。不需修改可跳过。
若镜像于服务器运行建议使用强密码,并在第一次登录后关闭密码登录,改用秘钥登录。
## 配置 Docker Compose
### Fork 仓库
Fork 本仓库,复制 Fork 后仓库的 HTTPS 地址,如 https://github.com/oceanbase/miniob.git.
也可以不 Fork,用本仓库 Git HTTPS 地址,后续自行进行 `git remote add`.
下面配置镜像中的仓库 URL 及克隆方式。
修改 **docker/docker-compose.yml** 文件的最后两行。注意等号前后不要有空格。
### Clone by SSH
`REPO_ADDR` 设置为 SSH URL, 如:git@github.com:oceanbase/miniob.git
### Clone by HTTPS
#### 私有仓库
如果 fork 出来的是私有仓库。
假设仓库 HTTPS 地址为:https://github.com/oceanbase/miniob.git.
**REPO_ADDR 格式**:https://\<username\>:\<password\>@github.com/oceanbase/miniob.git
`username``password` GitHub / Gitee(GitHub 不支持 password,需要创建 token,见[链接](https://docs.github.com/en/authentication/keeping-your-account-and-data-secure/creating-a-personal-access-token))。
**用户名如果是邮箱,字符 '@' 需要转义为 '%40'**.
```docker
# example
REPO_ADDR=https://oceanbase%40email.com:mypassword@github.com/oceanbase/miniob.git
```
#### 公有仓库
如果 fork 出的是公有仓库,直接将 `REPO_ADDR` 设置为仓库地址即可。
```Docker
REPO_ADDR=https://github.com/oceanbase/miniob.git
```
### 配置映射端口
默认映射本机端口 10000 到容器端口 22. 可在 docker-compose.yml 中自行修改。不许修改可跳过此步骤。
## 启动容器
### Clone by SSH
如果用 SSH 的方式 clone,需要将 Github/Gitee 对应的 SSH 私钥设置到环境变量内再启动 Container.
**假设 Github/Gitee 上传的公钥对应的秘钥位于 ~/.ssh/id_rsa.**
terminal 进入本仓库 docker 目录,在 docker 目录下执行命令启动容器。
#### Linux Shell
```bash
export PRIVATE_KEY=$(cat ~/.ssh/id_rsa) && docker-compose up -d --build
```
#### Windows Powershell
```bash
$env:PRIVATE_KEY=$(cat ~/.ssh/id_rsa) && docker-compose up -d --build
```
### Clone by HTTPS
不需要 PRIVATE_KEY 环境变量。
```bash
docker-compose up -d --build
```
此输出表示容器启动成功。
![container started](./images/vsc_container_started.png)
### 管理容器
一些管理容器的命令。
```bash
# 停止容器
docker stop miniob-dev
# 启动容器
docker start miniob-dev
# 重启容器
docker restart miniob-dev
```
## 使用 VSCode 远程开发
打开 VSCode,设置远程开发及调试。
### 安装 Remote-SSH 插件
一般会默认安装。快捷键组合 Ctrl + Shift + x 打开 VSCode 插件面板,检查 Remote - SSH 插件是否安装,如果没有安装就安装此插件。
![remote ssh](./images/vsc_remote_ssh_extension.png)
### 连接 Docker 容器
#### 配置 SSH Host
按快捷键 Ctrl + p, 输入 `remote-ssh: connect to host` 连接远程主机。
![connect to host](./images/vsc_remote_ssh_connect_cmd.png)
选择 `Add new ssh host`.
![add ssh host](./images/vsc_add_new_ssh_host.png)
输入命令.
```bash
ssh root@localhost -p 10000
```
以 root 用户连接容器的 10000 端口.
![ssh cmd](./images/vsc_ssh_connect.png)
选择写入哪个配置文件,默认第一个就行。
![ssh config ifle](./images/vsc_config_file.png)
#### 连接容器
按快捷键 Ctrl + p, 输入 `remote-ssh: connect to host` 连接远程主机。
主机选 localhost 并回车,跳出密码输入页面。若在之前步骤没有修改,密码默认为 root.
![password](./images/vsc_pwd.png)
### 打开代码目录
代码位于 /root/source/miniob. 点击左侧 `open folder`,进入该目录。需要再输入一次密码。
![open folder](./images/vsc_open_folder.png)
### 安装开发插件
Ctrl + Shift + x 打开 VSCode 插件面板, 安装如下插件. 注意要安装在 Docker 容器中,点击 `install in SSH:localhost`.
C/C++ 和 C/C++ Extension Pack.
![ext1](./images/vsc_cpp_extensions.png)
C/C++ Extension Pack 安装好后会让你选择 kits, 选 GCC 8.5 这个。
![kits](./images/vsc_kit_for_miniob.png)
CMake 和 CMake Tools.
![ext2](./images/vscode_search_plugs.png)
### 进行 Debug
#### 启动 Server
在 src/observer/sql/parser/parse_stage.cpp 的 handle_event 函数开头打个断点, 也就是第 90 行。
按 F5 开始调试。出现如下界面表示 miniob 的 server 端启动成功。
![server started](./images/vsc_server_started.png)
#### 启动 Client
Ctrl + Shift + ` 新启动一个 shell.
执行 `./build/bin/obclient` 启动客户端.
#### 执行 SQL
执行一句 SQL 测试断点是否正常工作。
```sql
show tables;
```
![debug](./images/vsc_debug.png)
正常进入断点。后续可以配置 Git 进行开发了。
\ No newline at end of file
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册