Docker Gitlab-CE HA(主备方式)

写在前面

  • 所有组件均使用容器方式
  • 目前仅仅做到了Gitlab-CE的主备方式高可用,没有做到双活

集群主机IP及组件说明

  • 192.168.26.101 [etcd,patroni,redis-cluster,redis-sentinel,gitlab]
  • 192.168.26.102 [etcd,patroni,redis-cluster,redis-sentinel,gitlab]
  • 192.168.26.103 [etcd,patroni,redis-cluster,redis-sentinel,gitlab]
  • 192.168.26.200 VIP地址用于patroni集群Master使用
  • 192.168.26.201 VIP地址用于gitlab-ce的外部url使用
  • 192.168.26.10 NFS地址
  • 执行循序 etcd -> patroni -> redis-cluster -> redis-sentinel -> mount /etc/fstab -> 等待gitlab自运行

postgres集群

etcd集群

//192.168.26.101
HOST_NAME=etcd-1
ETCD_ENDPOINT=etcd-1=http://192.168.26.101:2380,etcd-2=http://192.168.26.102:2380,etcd-3=http://192.168.26.103:2380
HOST_IP=`python -c "import socket;res = socket.gethostbyname(socket.gethostname());print(res)"`
ETCD_DATA=/opt/etcd/data

docker run -d --restart=always --net=host \
-v $ETCD_DATA:/etcd-data \
--name $HOST_NAME \
quay.io/coreos/etcd:v3.5.7 \
/usr/local/bin/etcd \
-name $HOST_NAME \
--data-dir /etcd-data \
--listen-client-urls http://0.0.0.0:2379 \
--advertise-client-urls http://0.0.0.0:2379 \
--listen-peer-urls http://0.0.0.0:2380 \
--initial-advertise-peer-urls http://$HOST_IP:2380 \
--initial-cluster "$ETCD_ENDPOINT" \
--initial-cluster-token tkn \
--initial-cluster-state new \
--log-level info \
--logger zap \
--log-outputs stderr

//192.168.26.102
HOST_NAME=etcd-2
ETCD_ENDPOINT=etcd-1=http://192.168.26.101:2380,etcd-2=http://192.168.26.102:2380,etcd-3=http://192.168.26.103:2380
HOST_IP=`python -c "import socket;res = socket.gethostbyname(socket.gethostname());print(res)"`
ETCD_DATA=/opt/etcd/data

docker run -d --restart=always --net=host \
-v $ETCD_DATA:/etcd-data \
--name $HOST_NAME \
quay.io/coreos/etcd:v3.5.7 \
/usr/local/bin/etcd \
-name $HOST_NAME \
--data-dir /etcd-data \
--listen-client-urls http://0.0.0.0:2379 \
--advertise-client-urls http://0.0.0.0:2379 \
--listen-peer-urls http://0.0.0.0:2380 \
--initial-advertise-peer-urls http://$HOST_IP:2380 \
--initial-cluster "$ETCD_ENDPOINT" \
--initial-cluster-token tkn \
--initial-cluster-state new \
--log-level info \
--logger zap \
--log-outputs stderr



//192.168.26.103
HOST_NAME=etcd-3
ETCD_ENDPOINT=etcd-1=http://192.168.26.101:2380,etcd-2=http://192.168.26.102:2380,etcd-3=http://192.168.26.103:2380
HOST_IP=`python -c "import socket;res = socket.gethostbyname(socket.gethostname());print(res)"`
ETCD_DATA=/opt/etcd/data

docker run -d --restart=always --net=host \
-v $ETCD_DATA:/etcd-data \
--name $HOST_NAME \
quay.io/coreos/etcd:v3.5.7 \
/usr/local/bin/etcd \
-name $HOST_NAME \
--data-dir /etcd-data \
--listen-client-urls http://0.0.0.0:2379 \
--advertise-client-urls http://0.0.0.0:2379 \
--listen-peer-urls http://0.0.0.0:2380 \
--initial-advertise-peer-urls http://$HOST_IP:2380 \
--initial-cluster "$ETCD_ENDPOINT" \
--initial-cluster-token tkn \
--initial-cluster-state new \
--log-level info \
--logger zap \
--log-outputs stderr

patroni集群

//192.168.26.101
HOST_NAME=patroni-1
ETCD_ENDPOINT=192.168.26.101:2379,192.168.26.102:2379,192.168.26.103:2379
PATRONI_PASSWD='123456'
PATRONI_VIP='192.168.26.200'
PATRONI_BRD='192.168.26.255'
PATRONI_DATA=/opt/postgres/data/

docker run -d --restart=always --name=$HOST_NAME \
--privileged \
--net=host \
--hostname=$HOST_NAME \
-e ETCD_ENDPION=$ETCD_ENDPOINT \
-e PATRONI_PASSWD=$PATRONI_PASSWD \
-e PATRONI_VIP=$PATRONI_VIP \
-e PATRONI_BRD=$PATRONI_BRD \
-v $PATRONI_DATA:/home/postgres/data/ \
swr.cn-north-4.myhuaweicloud.com/easyk8s.com/postgres_cluster:2023_02_07


//192.168.26.102
HOST_NAME=patroni-2
ETCD_ENDPOINT=192.168.26.101:2379,192.168.26.102:2379,192.168.26.103:2379
PATRONI_PASSWD='123456'
PATRONI_VIP='192.168.26.200'
PATRONI_BRD='192.168.26.255'
PATRONI_DATA=/opt/postgres/data/

docker run -d --restart=always --name=$HOST_NAME \
--privileged \
--net=host \
--hostname=$HOST_NAME \
-e ETCD_ENDPION=$ETCD_ENDPOINT \
-e PATRONI_PASSWD=$PATRONI_PASSWD \
-e PATRONI_VIP=$PATRONI_VIP \
-e PATRONI_BRD=$PATRONI_BRD \
-v $PATRONI_DATA:/home/postgres/data/ \
swr.cn-north-4.myhuaweicloud.com/easyk8s.com/postgres_cluster:2023_02_07


//192.168.26.103
HOST_NAME=patroni-3
ETCD_ENDPOINT=192.168.26.101:2379,192.168.26.102:2379,192.168.26.103:2379
PATRONI_PASSWD='123456'
PATRONI_VIP='192.168.26.200'
PATRONI_BRD='192.168.26.255'
PATRONI_DATA=/opt/postgres/data/

docker run -d --restart=always --name=$HOST_NAME \
--privileged \
--net=host \
--hostname=$HOST_NAME \
-e ETCD_ENDPION=$ETCD_ENDPOINT \
-e PATRONI_PASSWD=$PATRONI_PASSWD \
-e PATRONI_VIP=$PATRONI_VIP \
-e PATRONI_BRD=$PATRONI_BRD \
-v $PATRONI_DATA:/home/postgres/data/ \
swr.cn-north-4.myhuaweicloud.com/easyk8s.com/postgres_cluster:2023_02_07

验证集群

# 验证postgres_cluster集群
# docker exec -it patroni-[1/2/3] bash
$ patronictl -c /home/postgres/config/patroni.yml list

redis 哨兵集群

redis-cluster

//192.168.26.101
#!/bin/bash
HOST_NAME=redis_server-1
REDIS_DATA=/opt/redis/data

docker run -d --name $HOST_NAME \
--net=host \
-v $REDIS_DATA:/data \
redis:latest  redis-server --port 6379

//192.168.26.102
#!/bin/bash
HOST_NAME=redis_server-2
REDIS_DATA=/opt/redis/data
REDIS_MASTER_IP='192.168.26.101'
REDIS_MASTER_PORT=6379

docker run -d --name $HOST_NAME \
--net=host \
-v /opt/redis/data:/data \
redis:latest redis-server --slaveof $REDIS_MASTER_IP $REDIS_MASTER_PORT --port 6379

//192.168.26.103
#!/bin/bash
HOST_NAME=redis_server-3
REDIS_DATA=/opt/redis/data
REDIS_MASTER_IP='192.168.26.101'
REDIS_MASTER_PORT=6379

docker run -d --name $HOST_NAME \
--net=host \
-v /opt/redis/data:/data \
redis:latest redis-server --slaveof $REDIS_MASTER_IP $REDIS_MASTER_PORT --port 6379

redis-sentinel

//192.168.26.101
REDIS_NAME_IP='192.168.26.101'
REDIS_NAME_PORT=6379
REDIS_SENTINEL_CONFIG=/opt/redis/sentinel.conf
REDIS_SENTINEL_PORT=5000
HOST_NAME=reids_sentinel-1

if [[ ! -e ${REDIS_SENTINEL_CONFIG}gitlab.rb ]];
then

cat > $REDIS_SENTINEL_CONFIG <<EOF
protected-mode no
bind 0.0.0.0
port 5000
daemonize no
sentinel monitor mymaster $REDIS_NAME_IP $REDIS_NAME_PORT 2
sentinel down-after-milliseconds mymaster $REDIS_SENTINEL_PORT
sentinel failover-timeout mymaster 180000
sentinel parallel-syncs mymaster 1
EOF

fi

docker run -d --restart=always --name $HOST_NAME \
-p $REDIS_SENTINEL_PORT:$REDIS_SENTINEL_PORT \
-v /opt/redis/sentinel.conf:/etc/redis/sentinel.conf \
redis redis-sentinel /etc/redis/sentinel.conf


//192.168.26.102
REDIS_NAME_IP='192.168.26.101'
REDIS_NAME_PORT=6379
REDIS_SENTINEL_CONFIG=/opt/redis/sentinel.conf
REDIS_SENTINEL_PORT=5000
HOST_NAME=reids_sentinel-2

if [[ ! -e ${REDIS_SENTINEL_CONFIG}gitlab.rb ]];
then

cat > $REDIS_SENTINEL_CONFIG <<EOF
protected-mode no
bind 0.0.0.0
port 5000
daemonize no
sentinel monitor mymaster $REDIS_NAME_IP $REDIS_NAME_PORT 2
sentinel down-after-milliseconds mymaster $REDIS_SENTINEL_PORT
sentinel failover-timeout mymaster 180000
sentinel parallel-syncs mymaster 1
EOF

fi

docker run -d --restart=always --name $HOST_NAME \
-p $REDIS_SENTINEL_PORT:$REDIS_SENTINEL_PORT \
-v /opt/redis/sentinel.conf:/etc/redis/sentinel.conf \
redis redis-sentinel /etc/redis/sentinel.conf


//192.168.26.103
REDIS_NAME_IP='192.168.26.101'
REDIS_NAME_PORT=6379
REDIS_SENTINEL_CONFIG=/opt/redis/sentinel.conf
REDIS_SENTINEL_PORT=5000
HOST_NAME=reids_sentinel-3

if [[ ! -e ${REDIS_SENTINEL_CONFIG}gitlab.rb ]];
then

cat > $REDIS_SENTINEL_CONFIG <<EOF
protected-mode no
bind 0.0.0.0
port 5000
daemonize no
sentinel monitor mymaster $REDIS_NAME_IP $REDIS_NAME_PORT 2
sentinel down-after-milliseconds mymaster $REDIS_SENTINEL_PORT
sentinel failover-timeout mymaster 180000
sentinel parallel-syncs mymaster 1
EOF

fi

docker run -d --restart=always --name $HOST_NAME \
-p $REDIS_SENTINEL_PORT:$REDIS_SENTINEL_PORT \
-v /opt/redis/sentinel.conf:/etc/redis/sentinel.conf \
redis redis-sentinel /etc/redis/sentinel.conf

验证Redis集群

# pip install redis -i http://mirrors.aliyun.com/pypi/simple 

# cat redis_test.py
# -*- coding:utf-8 -*-  
from redis.sentinel import Sentinel

# 连接哨兵服务器(主机名也可以用域名)
sentinel = Sentinel([('192.168.26.101', 5000),
                     ('192.168.26.102', 5001),
                     ('192.168.26.103', 5001)
                     ],
                    socket_timeout=0.5)

# 获取主服务器地址
master = sentinel.discover_master('mymaster')
print("redis master ip",master)

slave = sentinel.discover_slaves('mymaster')
print("redis slave ip",slave)

master = sentinel.master_for('mymaster', socket_timeout=0.5, db=15)
w_ret = master.set('foo', 'bar')

slave = sentinel.slave_for('mymaster', socket_timeout=0.5, db=15)
r_ret = slave.get('foo')
print(r_ret)

# python3 redis.py
redis master ip ('192.168.26.101', 6379)
redis slave ip [('192.168.26.102', 6379), ('192.168.26.103', 6379)]
b'bar'

redis create user And passwd And tables

// 需要取Master主机执行,也就是挂载192.168.26.200的机器
# docker exec -it patroni-3 bash
$ psql
$ create role gitlab login encrypted password 'pass';
$ create database gitlabhq_production owner=gitlab ENCODING = 'UTF8';
$ \c gitlabhq_production
$ CREATE EXTENSION pg_trgm;
$ select extname,extowner,extnamespace,extrelocatable,extversion from pg_extension;

Gitlab-CE

mount /etc/fstab
//[注意]NFS的设置sync,no_root_squash,no_all_squash

//192.168.26.101
# mkdir -p /opt/gitlab/{config,data,master}
# vi /etc/fstab
192.168.26.10:/data/gitlab/config /opt/gitlab/config nfs4 defaults        0 0
192.168.26.10:/data/gitlab/data /opt/gitlab/data nfs4 defaults        0 0
192.168.26.10:/data/gitlab/master /opt/gitlab/master nfs4 defaults        0 0
# mount -a

//[注意要设置每台机器的Crontab]
# crontab -l
# */3 * * * * /bin/bash /opt/gitlab.sh > /dev/null 2>&1

# vi /opt/gitlab.sh
#!/bin/bash

# 配合crontab 主备
# 注意MASTER_DIR 为NFS共享目录用于抢占写入
MASTER_SEZIE_CONFIG=/opt/gitlab/master/sezie.config
# 注意GIT_CONFIG/DATA为NFS共享目录用于配置文件和数据目录
GIT_CONFIG=/opt/gitlab/config/
GIT_DATA=/opt/gitlab/data/

# [注意]GIT配置文件 
# 需要创建PG中的数据库,用户名,密码,库
# 配置文件中redis的哨兵地址

if [[ ! -e ${GIT_CONFIG}gitlab.rb ]];
then

cat > ${GIT_CONFIG}gitlab.rb << EOF
external_url 'http://192.168.26.201'
postgresql['enable'] = false
gitlab_rails['db_adapter'] = "postgresql"
gitlab_rails['db_encoding'] = "utf8"
gitlab_rails['db_database'] = "gitlabhq_production"
gitlab_rails['db_pool'] = 30
gitlab_rails['db_username'] = "gitlab"
gitlab_rails['db_password'] = "pass"
gitlab_rails['db_host'] = "192.168.26.200"
gitlab_rails['db_port'] = "5432"

redis['enable'] = false
gitlab_rails['redis_sentinels'] = [
{'host' => '192.168.26.101', 'port' => 5000},
{'host' => '192.168.26.102', 'port' => 5000},
{'host' => '192.168.26.103', 'port' => 5000},
]
redis['master_name'] = 'mymaster'
gitlab_rails['redis_database'] = 0
EOF

fi

# 最大超时时间
TIME_MAX=300


# 抢占启动模块
function start_gitlab(){
  echo "开启抢占"
  IP=`python -c "import socket;res = socket.gethostbyname(socket.gethostname());print(res)"`
  TIME=`date "+%s"`
  echo -n "$IP $TIME" > $MASTER_SEZIE_CONFIG
  docker ps -a | grep gitlab > /dev/null 2>&1
  if [[ $? -eq 0 ]];
  then
    docker start gitlab
  else
docker run -d \
--name gitlab \
-p 443:443 -p 80:80 -p 2222:22 \
-v $GIT_CONFIG:/etc/gitlab/ \
-v /opt/gitlab/log/:/var/log/gitlab \
-v $GIT_DATA:/var/opt/gitlab/ \
gitlab/gitlab-ce
  fi
}

# 判断服务是否是本地提供,不是停止
function stop_gitlab(){
  IP=`cat $MASTER_SEZIE_CONFIG | awk '{print $1}'`
  LOCAL_IP=`python -c "import socket;res = socket.gethostbyname(socket.gethostname());print(res)"`
  if [[ $LOCAL_IP != $IP ]]; 
  then 
    docker stop gitlab
  fi
}

if [[ -e $MASTER_SEZIE_CONFIG ]];
then
  # 存在
  echo "文件存在"
  IP=`cat $MASTER_SEZIE_CONFIG | awk '{print $1}'`
  TIME=`cat $MASTER_SEZIE_CONFIG | awk '{print $2}'`
  HTTP_CODE=`curl -sI http://$IP/users/sign_in | head -n 1 | awk '{print $2}'`
  END_TIME=`date "+%s"`
  if [[ $HTTP_CODE -eq 200 ]];
  then
    # 访问成功更新时间戳
    echo -n "$IP $END_TIME" > $MASTER_SEZIE_CONFIG
    stop_gitlab
  else
    # 访问失败判断时间戳差额
    TIME_TMP=$(( END_TIME -  TIME ))
    if [[ $TIME_TMP -gt $TIME_MAX ]];
    then
      # 大于超时开始抢占
      start_gitlab
    fi
  fi
else
  # 不存在
  echo "文件不存在"
  start_gitlab
fi

验证环节

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,525评论 6 507
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,203评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,862评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,728评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,743评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,590评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,330评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,244评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,693评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,885评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,001评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,723评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,343评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,919评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,042评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,191评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,955评论 2 355

推荐阅读更多精彩内容