본문 바로가기

엔지니어/DB

Maria DB Galera Cluster

728x90
반응형

MariaDB에서 MariaDB/Galera Cluster 제품군을 새롭게 출시하였습니다.MariaDB/Galera는 MariaDB의 Synchronous 방식으로 동작하는 다중 마스터 클러스터입니다.

MariaDB/Galera Cluster은 Galera 라이브러리를 사용하여 노드 간 데이터 복제를 수행합니다. 물론 아직은 Alpha 버전으로 발표되기는 했지만, 조만간 안정적인 버전이 릴리즈 되면 상당한 물건이 될만한 놈입니다.

오늘은 이에 관해 간단하게 리뷰를 하겠습니다.

Feature & Benefits

먼저 MariaDB/Galera Cluster의 특징은 다음과 같이 몇 가지로 나눠볼 수 있습니다.

  • Synchronous 방식으로 노드 간 데이터 복제
  • Active-Active 방식의 다중 마스터 구성 – 모든 노드에서 읽기/쓰기가 가능
  • 클러스터 내 노드 자동 컨트롤 – 특정 노드 장애 시 자동으로 해당 노드 제거
  • 자동으로 신규 노드 추가
  • 완벽한 병렬적으로 데이터를 행단위로 복제
  • 기존의 MySQL 클라이언트 방식으로 동작

cluster-diagram1
출처 : http://www.percona.com/doc/percona-xtradb-cluster/_images/cluster-diagram1.png

이와 같은 특징에서 전통적인 Asynchronous 방식의 리플리케이션이 가지는 한계점이 해결됩니다.

  • 마스터/슬레이브간 데이터 동기화 지연 없음
  • 노드 간 유실되는 트랜잭션이 없음
  • 읽기/쓰기 모두 확장이 가능
  • 클라이언트의 대기 시간이 줄어듬 – 데이터는 로컬 노드에 존재

하지만 Replication이 가지는 본연의 한계점은 여전히 내재합니다.

  • 신규 노드 추가 시 부하 발생 – 신규 노드 추가 시 모든 데이터를 복사해야 함
  • 효과적인 쓰기 확장 솔루션에는 한계 – 서버 간 Group Communication시 트래픽 발생
  • 모든 서버 노드에 동일한 데이터를 유지해야 함 – 저장 공간 낭비

MariaDB/Galera Cluster

MariaDB/Galera cluster는 Galera 라이브러리를 사용하여 리플리케이션을 수행한다고 하는데 Galera Replication은 어떤 방식으로 동작할까요?

Galera Replication은 wsrep API로 노드 간 통신을 하며, MariaDB 측에서는 wsrep API에 맞게 내부적인 개선하였다고 합니다. MySQL-wsrep는 https://launchpad.net/codership-mysql에서 시작한 오픈소스 프로젝트입니다.

MySQL-wsrep는 MySQL의 InnoDB스토리지 엔진 내부에서 Write Set(기록 집합 : 트랜잭션의 기록하는 모든 논리적인 데이터 집합)을 추출하고 적용하는 구현됩니다. 노드 간 Write Set을 전송 및 통신을 위해서는 별도의 리플리케이션 플러그인을 사용하며, 리플리케이션 엔진은 wsrep에 정의된 Call/Callback 함수에 따라 동작합니다.

1) Synchronous vs. Asynchronous

먼저 Synchronous와 Asynchronous 리플리케이션의 차이점에 대해서 설명하겠습니다.

리플리케이션의 두 가지 방식의 가장 기본적인 차이점은 클러스터 내 특정 노드에서 데이터 변경이 발생하였을 때 다른 노드들에 동시에 데이터 변경이 적용되는 것을 보장는지 여부에 있습니다.

Asynchronous 방식의 Replication은 마스터 노드에서 발생한 변화가 슬레이브 노드에 동시에 적용되는 것을 보장하지 않습니다. 마스터/슬레이브 간 데이터 동기화 지연은 언제든 발생할 수 있으며, 마스터 노드가 갑자기 다운되는 경우 가장 최근의 변경 사항이 슬레이브 노드에서는 일부 유실될 수도 있는 가능성도 있습니다.

Synchronous 방식의 Replication은 Asynchronous에 비해 다음과 같은 강점을 가집니다.

  • 노드 장애 시에도 데이터 유실 없이 높은 가용성 달성
  • 트랜잭션은 모든 노드에서 동시 다발적으로 발생
  • 클러스트 내 모든 노드 간 데이터 일관성을 보장

그러나 Synchronous Replication 수행을 위해서는 2단계 Commit 필요하거나 분산 잠금과 같은 상당히 느린 방식으로 동작합니다.

Synchronous 방식의 Replication은 성능 이슈와 및 복잡한 구현 내부적으로 요구되기 때문에 여전히 Asynchronous 방식의 Replication이 널리 사용되고 있습니다. 오픈 소스 대명사로 불리는 MySQL과 PostgreSQL이 Asynchronous 방식으로만 데이터 복제가 이루어지는 것 또한 그와 같은 이유에서입니다.

2) Certification Based Replication Method

성능 저하 없이 Synchronous하게 데이터베이스 리플리케이션을 구현하기 위해 “Group Communication and Transaction Ordering techniques”이라는 새로운 방식이 고안되었습니다. 이것은 많은 연구자들(Database State Machine Approach and Don’t Be Lazy, Be Consistent)이 제안했던 방식으로, 프로토타입 구현해본 결과 상당한 발전 가능성을 보여주었던 바가 있습니다.

Galera Replication은 높은 가용성과 성능이 필요한 어플리케이션에서는 상당히 쉽고 확장 가능한 Synchronous 방식의 리플리케이션을 제공하며 다음과 같은 특징이 있습니다.

  • 높은 가용성
  • 높은 투명성(알기 쉽다는 의미)
  • 높은 확장성(어플리케이션에 따라 거의 선형적인 확장까지도 가능)

Galera 리플리케이션은 분할된 라이브러리와 같이 동작하고, wsrep API로 동작하는 시스템이라면 어떠한 트랜잭션과도 연관되어 동작할 수 있는 구조입니다.

3) Galera Replication implementation

Galera Replication의 가장 큰 특징은 트랜잭션이 커밋되는 시점에 다른 노드에 유효한 트랜잭션인지 여부를 체크하는 방식으로 동작한다는 점입니다. 클러스트 내에서 트랜잭션은 모든 서버에 동시에 반영되거나 전부 반영되지 않는 경우 둘 중 하나입니다.

트랜잭션 커밋이 가능한 여부는 네트워크를 통해서 다른 노드와의 통신에서 결정합니다. 그렇기 때문에 커밋 시 커밋 요청에 대한 응답 시간이 존재하죠. 커밋 요청에 대한 응답 시간은 다음 요소에 영향을 받습니다.

  • 네트워크 왕복 시간
  • 타 노드에서 유효성 체크 시간
  • 각 노드에서 데이터 반영 시간

여기서 재미있는 사실은 트랜잭션을 시작하는 시점(BEGIN)에는 자신의 노드에서는 Pessimistic Locking으로 동작하나, 노드 사이에서는 Optimistic Locking Model로 동작한다는 점입니다. 먼저 트랜잭션을 자신의 노드에 수행을 하고, 커밋을 한 시점에 다른 노드로부터 해당 트랜잭션에 대한 유효성을 받는다는 것이죠. 보통 InnoDB와 같이 트랜잭션을 지원하는 시스템인 경우 SQL이 시작되는 시점에서 Lock 감지를 하나, 여기서는 커밋되는 시점에 노드 간 트랜잭션 유효성 체크를 합니다.

위 그림에서 커밋되는 시점에 마스터 노드에서 슬레이브 노드에 이벤트를 날립니다. 그리고 슬레이브 노드에서 유효성 체크 후 데이터를 정상적으로 반영하게되면 실제 마스터 노드에서 커밋 완료가 되는 것이죠. 그렇지 않은 경우는 롤백 처리됩니다. 이 모든 것은 클러스터 내부에서 동시에 이루어집니다.

4) Galera Replication VS MySQL Replication

CAP 모델 관점에서 본다면 MySQL Replication은 “Availability”과 “Partitioning tolerance”로 동작하지만 Galera Replication에서는  “Consistency”과 “Availability” 로 동작한다는 점에서 차이가 있습니다. MySQL Replication은 데이터 일관성을 보장하지 않음에 반해 Galera Replication은 데이터 일관성을 보장합니다.

C – Consistency (모든 노드의 데이터 일관성 유지)
A – Availability (모든 노드에서 항시 Read/Write이 가능해야 함)
P – Partitioning tolerance (내부 네트워크 단절 시에도 정상적으로 작동해야함)

5) Limitations

현재는 Alpha 버전으로 릴리즈되었고, 추후 안정적인 버전이 나오겠지만, 태생적으로 가지는 한계가 있습니다.

데이터 일관성 유지를 위해서 트랜잭션이 필요한 만큼, 트랜잭션이 기능이 있는 스토리지 엔진에서만 동작합니다. 현재까지는 오직 InnoDB에서만 가능하다고 하네요. MyISAM과 같이 커밋/롤백 개념이 없는 스토리지 엔진은 데이터 복제가 이뤄질 수 없다는 점이죠.

Row 기반으로 데이터 복제가 이루어지기 때문에 반드시 Primary Key가 있어야 합니다. Oracle RAC와 같이 공유 스토리지에서 동일한 데이터 파일을 사용한다면 Rowid가 같으므로 큰 문제가 없겠지만, 물리적으로 스토리지가 독립적인 구조이기 때문이죠. 이것은 기존 MySQL Replication에서도 주의하고 사용해야할 사항이기도 합니다.

최대 가능한 트랜잭션 사이즈는 wsrep_max_ws_rows와 wsrep_max_ws_size에 정의된 설정 값에 제약을 받으며, LOAD DATA INFILE 처리 시 매 1만 건 시 커밋이 자동으로 이루어집니다.

트랜잭션 유효성이 커밋되는 시점에서 이루어지며, 동일한 행에 두 개의 노드에서 데이터 변경을 시도한다면 오직 하나의 노드에서만 데이터 변경이 가능합니다.

또한 원거리 리플리케이션 경우 커밋에 대한 응답 요청으로 인하여 전반적인 시스템 성능 저하가 발생합니다.

Conclusion

MariaDB/Galera Cluster은 전통적인 MySQL Replication이 가지는 가장 큰 문제점이었던 데이터 동기화 지연과 노드 간 트랜잭션 유실 가능에 대한 해결책을 제시합니다.

또한 노드 내부에서는 InnoDB 고유의 트랜잭션으로 동작하고, 실제 커밋이 발생하는 시점에 다른 노드에게 유효성을 체크 및 동시 커밋한다는 점에서 재미있는 방식으로 동작하죠. 결국 기존 MySQL 아키텍트는 그대로 유지하고, Replication 동작에 관한 방법만 수정하여 RDBMS 기반의 분산 DBMS 를 내놨다는 점에서 상당히 흥미로운 제품입니다.

그러나, 동일한 데이터 변경 이슈가 많은 서비스 경우 노드 간 데이터 충돌이 자주 발생 가능성이 있을 것으로 판단됩니다. 데이터 충돌이 발생하여 자주 트랜잭션 롤백이 발생하면 사용자 별로 원활한 서비스 사용이 불가하니, 이에 대한 대책을 어플리케이션 레벨에서 적절하게 고려하여 서비스 설계를 해야 하겠죠. 예를 들어 노드 단위로 주로 변경할 데이터를 나눠서 처리하는 방식으로 서비스 설계가 이뤄져야하지 않을까 생각합니다.

Synchronous 방식으로 노드 간 데이터 복제가 이루어진다는 점에서 아주 반가운 소식이기는 하지만, 기존과 같이 데이터를 설계하면 오히려 서비스 안정성이 크게 떨어질 수도 있다는 점에서 새로운 변화가 예상됩니다.





설정방법


-- 아래의 작업은 모두 root 계정으로 진행한다. 

-- 아래 명령어에서 sudo 는 root 권한이 있는 일반 계정으로 명령어 실행할 때 사용하는 것인데,

-- root 로 진행한다면 제거해도 된다.


환경 :

maria01 : 10.17.132.101, CentOS 6.4

maria02 : 10.17.132.102, CentOS 6.4

maria03 : 10.17.132.103, CentOS 6.4



-- 일반적으로 필요한 rpm 을 설치한다.(옵션)

yum install cmake

yum groupinstall 'Development Tools'


-- O.S의 SELinux 기능 해제 및 iptables 방화벽에서 포트 오픈

sudo setenforce 0


sudo lokkit --port 3306:tcp

sudo lokkit --port 4444:tcp

sudo lokkit --port 4567:tcp

sudo lokkit --port 4568:tcp


-- yum 으로 설치 시, 참조할 수 있는 URL 저장.

sudo cat >> /etc/yum.repos.d/MariaDB.repo << EOF

[mariadb]

name = MariaDB

baseurl = http://yum.mariadb.org/10.0/centos6-amd64

gpgkey=https://yum.mariadb.org/RPM-GPG-KEY-MariaDB

gpgcheck=1

EOF


-- EPEL 설치, yum 실행 시 오류 처리 

-- (아래 오류는 처음 yum 실행 시 발생할 수 도 있고, 중간에 발생할 수도 있다.) 

sudo yum install http://dl.fedoraproject.org/pub/epel/6/x86_64/socat-1.7.2.3-1.el6.x86_64.rpm


or


sudo yum install epel-release

sudo yum install socat

실행 시 "Error : Cannot retrieve metalink for repositroy: epel. Please verify its path and try again" 발생


/etc/yum.repos.d/epel.repo 

metalink를 참조하는 mirrorlist를 주석 처리하고, 

그 위의 baseurl을 그대로 사용하도록 주석 해제한다.

enabled은 1로 설정


-- MariaDB Galera Server Cluster, Client, Rsync, Galera 설치가 되는데, yum 으로만 설치가 가능하고,

-- yum 설치 시, 설치되는 O.S에 맞게 다운로드해서 설치해준다. 

sudo yum install MariaDB-Galera-server MariaDB-client rsync galera


-- 각 DB 에서 MariaDB 를 시작한다.

sudo service mysql start


-- 각 DB 에서 아래 경로와 같이 명령어 실행

sudo /usr/bin/mysql_secure_installation


-->

Set root password? [Y/n] y

New password: 

Re-enter new password: 

Password updated successfully!

Reloading privilege tables..

 ... Success!



By default, a MariaDB installation has an anonymous user, allowing anyone

to log into MariaDB without having to have a user account created for

them.  This is intended only for testing, and to make the installation

go a bit smoother.  You should remove them before moving into a

production environment.


Remove anonymous users? [Y/n] y

 ... Success!


Normally, root should only be allowed to connect from 'localhost'.  This

ensures that someone cannot guess at the root password from the network.


Disallow root login remotely? [Y/n] n

 ... skipping.


By default, MariaDB comes with a database named 'test' that anyone can

access.  This is also intended only for testing, and should be removed

before moving into a production environment.


Remove test database and access to it? [Y/n] y

 - Dropping test database...

 ... Success!

 - Removing privileges on test database...

 ... Success!


Reloading the privilege tables will ensure that all changes made so far

will take effect immediately.


Reload privilege tables now? [Y/n] y

 ... Success!


Cleaning up...


All done!  If you've completed all of the above steps, your MariaDB

installation should now be secure.


Thanks for using MariaDB!

------------------------------------------------------------------------


-- 각 DB에서 아래와 같이 실행한다. root 패스워드 설정 및 Cluster를 운영할 계정 설정

mysql -u root -p

mysql> DELETE FROM mysql.user WHERE user='';

mysql> GRANT ALL ON *.* TO 'root'@'%' IDENTIFIED BY 'rootpassword';

mysql> GRANT USAGE ON *.* to clustuser@'%' IDENTIFIED BY 'clustuserpassword';

mysql> GRANT ALL PRIVILEGES on *.* to clustuser@'%';

mysql> FLUSH PRIVILEGES;

mysql> quit


-- 지금까지 설정된 사항 이후, 각 서버의 DB 를 중지한다.

sudo service mysql stop


-- 호스트명 maria01 에서 아래와 같이 파일을 생성한다.

sudo cat >> /etc/my.cnf.d/server.cnf << EOF

binlog_format=ROW

default-storage-engine=innodb

innodb_autoinc_lock_mode=2

innodb_locks_unsafe_for_binlog=1

query_cache_size=0

query_cache_type=0

bind-address=0.0.0.0


datadir=/data/R2U

innodb_log_file_size=100M

innodb_file_per_table

innodb_flush_log_at_trx_commit=2


wsrep_provider=/usr/lib64/galera/libgalera_smm.so

wsrep_cluster_address="gcomm://10.17.132.101,10.17.132.102,10.17.132.103"

wsrep_cluster_name='R2UDBCLUSTER'

wsrep_node_address='10.17.132.101'

wsrep_node_name='maria01'

wsrep_sst_method=rsync

wsrep_sst_auth=clustuser:clustuserpassword


EOF


-- 호스트명 maria02 에서 아래와 같이 파일을 생성한다.

sudo cat >> /etc/my.cnf.d/server.cnf << EOF

binlog_format=ROW

default-storage-engine=innodb

innodb_autoinc_lock_mode=2

innodb_locks_unsafe_for_binlog=1

query_cache_size=0

query_cache_type=0

bind-address=0.0.0.0


datadir=/data/R2U

innodb_log_file_size=100M

innodb_file_per_table

innodb_flush_log_at_trx_commit=2


wsrep_provider=/usr/lib64/galera/libgalera_smm.so

wsrep_cluster_address="gcomm://10.17.132.101,10.17.132.102,10.17.132.103"

wsrep_cluster_name='R2UDBCLUSTER'

wsrep_node_address='10.17.132.102'

wsrep_node_name='maria02'

wsrep_sst_method=rsync

wsrep_sst_auth=clustuser:clustuserpassword


EOF


-- 호스트명 maria03 에서 아래와 같이 파일을 생성한다.

sudo cat >> /etc/my.cnf.d/server.cnf << EOF

binlog_format=ROW

default-storage-engine=innodb

innodb_autoinc_lock_mode=2

innodb_locks_unsafe_for_binlog=1

query_cache_size=0

query_cache_type=0

bind-address=0.0.0.0


datadir=/data/R2U

innodb_log_file_size=100M

innodb_file_per_table

innodb_flush_log_at_trx_commit=2


wsrep_provider=/usr/lib64/galera/libgalera_smm.so

wsrep_cluster_address="gcomm://10.17.132.101,10.17.132.102,10.17.132.103"

wsrep_cluster_name='R2UDBCLUSTER'

wsrep_node_address='10.17.132.103'

wsrep_node_name='maria03'

wsrep_sst_method=rsync

wsrep_sst_auth=clustuser:clustuserpassword


EOF


-- 호스트명 maria01 에서 아래 명령어 실행한다.

sudo /etc/init.d/mysql start --wsrep-new-cluster


/var/lib/mysql/maria01.err 파일을 열어서 Error 가 있는지 확인한다.


-- 여기까지 정상적이면, 나머지는 다른 DB 시스템에서 아래와 같이 명령어를 실행한다.

sudo service mysql start


-- maria01 또는 다른 노드에서 아래 명령어를 실행 및 패스워드 입력해서 정상적으로 Clustering 됐는지 확인한다.

mysql -u root -p -e "show status like 'wsrep%'"


+------------------------------+----------------------------------------------------------+

| Variable_name                | Value                                                    |

+------------------------------+----------------------------------------------------------+

| wsrep_local_state_uuid       | 45d51cee-b18b-11e4-bf88-63f0c76b2a3e                     |

| wsrep_protocol_version       | 5                                                        |

| wsrep_last_committed         | 10013                                                    |

| wsrep_replicated             | 10011                                                    |

| wsrep_replicated_bytes       | 2572923                                                  |

| wsrep_repl_keys              | 30026                                                    |

| wsrep_repl_keys_bytes        | 470461                                                   |

| wsrep_repl_data_bytes        | 1461758                                                  |

| wsrep_repl_other_bytes       | 0                                                        |

| wsrep_received               | 92                                                       |

| wsrep_received_bytes         | 2022                                                     |

| wsrep_local_commits          | 10005                                                    |

| wsrep_local_cert_failures    | 0                                                        |

| wsrep_local_replays          | 0                                                        |

| wsrep_local_send_queue       | 0                                                        |

| wsrep_local_send_queue_avg   | 0.000000                                                 |

| wsrep_local_recv_queue       | 0                                                        |

| wsrep_local_recv_queue_avg   | 0.000000                                                 |

| wsrep_local_cached_downto    | 1                                                        |

| wsrep_flow_control_paused_ns | 0                                                        |

| wsrep_flow_control_paused    | 0.000000                                                 |

| wsrep_flow_control_sent      | 0                                                        |

| wsrep_flow_control_recv      | 0                                                        |

| wsrep_cert_deps_distance     | 64.164786                                                |

| wsrep_apply_oooe             | 0.000000                                                 |

| wsrep_apply_oool             | 0.000000                                                 |

| wsrep_apply_window           | 1.000000                                                 |

| wsrep_commit_oooe            | 0.000000                                                 |

| wsrep_commit_oool            | 0.000000                                                 |

| wsrep_commit_window          | 1.000000                                                 |

| wsrep_local_state            | 4                                                        |

| wsrep_local_state_comment    | Synced                                                   |--> Sync 여부

| wsrep_cert_index_size        | 61                                                       |

| wsrep_causal_reads           | 0                                                        |

| wsrep_cert_interval          | 0.000000                                                 |

| wsrep_incoming_addresses     | 10.17.132.101:3306,10.17.132.102:3306,10.17.132.103:3306 |--> 연결된 노드 확인

| wsrep_cluster_conf_id        | 4                                                        |

| wsrep_cluster_size           | 3                                                        |--> 연결된 노드 수

| wsrep_cluster_state_uuid     | 45d51cee-b18b-11e4-bf88-63f0c76b2a3e                     |

| wsrep_cluster_status         | Primary                                                  |

| wsrep_connected              | ON                                                       |--> 연결여부 ON 확인

| wsrep_local_bf_aborts        | 0                                                        |

| wsrep_local_index            | 0                                                        |

| wsrep_provider_name          | Galera                                                   |

| wsrep_provider_vendor        | Codership Oy <info@codership.com>                        |

| wsrep_provider_version       | 25.3.5(rXXXX)                                            |

| wsrep_ready                  | ON                                                       |--> 운영 여부 ON 확인

| wsrep_thread_count           | 2                                                        |

+------------------------------+----------------------------------------------------------+

위의 많은 내용 중 꼭 체크해야 할 것을 정리한다면, 

| wsrep_local_state_comment    | Synced                    |--> Sync 여부

| wsrep_incoming_addresses     | 10.17.132.101:3306,10.17.132.102:3306,10.17.132.103:3306 |

--> 연결된 노드 확인

| wsrep_cluster_size               | 3                             |--> 연결된 노드 수

| wsrep_connected                | ON                          |--> 연결여부 ON 확인

| wsrep_ready                      | ON                         |--> 운영 여부 ON 확인



반응형

'엔지니어 > DB' 카테고리의 다른 글

mysql bin-log(바이너리 로그) 관리  (0) 2017.07.07
oracle 10g 설치  (0) 2017.01.17
Mysql replication chain 제거 및 Master 변경  (0) 2017.01.17
MySQL 리플리케이션 상태 확인  (0) 2017.01.17
Mysql 5.7.14 설치 및 replication  (0) 2017.01.13