美文网首页
Docker Gitlab-CE HA(主备方式)

Docker Gitlab-CE HA(主备方式)

作者: 生活就是闹剧 | 来源:发表于2023-02-15 15:29 被阅读0次

写在前面

  • 所有组件均使用容器方式
  • 目前仅仅做到了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

验证环节

相关文章

网友评论

      本文标题:Docker Gitlab-CE HA(主备方式)

      本文链接:https://www.haomeiwen.com/subject/rkqukdtx.html