◎ Linux 가상 서버(LVS:Linux Virtual Server)

Linux 가상 서버(LVS:Linux Virtual Server)는 Linux에서 제공하는 L4 Load Balancer 솔루션입니다.

LVS는 크게 Packet Load Balacing을 수행하는 Load Balancer와 Packet의 실제 목적지인 Real Server로 구성됩니다.

여러 서버에 네트워크 트래픽을 분산하여 성능을 개선하고 고가용성을 보장하며 개별 시스템의 과부하를 방지하기 위해 널리 사용되는 강력한 로드 밸런싱 솔루션입니다. LVS는 OSI 모델의 전송 계층(Transport Layer, Layer 4)에서 작동하며 다양한 부하 분산 알고리즘을 사용하여 들어오는 요청을 백엔드 서버 풀에 효율적으로 분산합니다.

 

◎ IPVS(IP Virtual Server)
IPVS는 LVS 위에서 동작하는 L4 Load Balancer입니다.

 

◎ 구성도

[AS-IS]

WEB1과 WEB2는 동기화 설정이 되어있어 어느 서버에 접속하더라도 동일한 내용을 보여주고 있으나 부하분산(LB) 구성이 되어 있지않아 특정 서버로만 접속이 몰리는 현상이 발생할 수 있습니다.

[TO-BE]

IPVS를 구성하여 부하를 분산해줍니다. IPVS를 구성할 때는 IP를 총 2개(Real IP와 Virtual IP) 할당해 주어야 합니다.

Virtual IP는 부하분산(LB)용이고, Real IP는 LVS 관리용 IP입니다. Real IP 1개로는 IPVS 구성이 불가합니다.

 

 구성방법

※ CentOS를 기준으로 테스트가 완료된 설정방법입니다.

[공통]

WEB1, WEB2 및 LVS 서버에 아래 명령어를 실행합니다.

> 네트워크 IP 설정(생략)

> DNS IP 설정

# vi /etc/resolv.conf
nameserver 168.126.63.1

vi /etc/resolv.conf : DNS 서버 설정 파일입니다.

> 방화벽 규칙 설정
들어오는 트래픽이 로드 밸런서에 도달하고 백엔드 서버에서 나가는 트래픽이 통과할 수 있도록 방화벽 규칙이 설정되어 있는지 확인합니다. 보통은 방화벽 장비에서 정책으로 통제하기 때문에 서버에서 방화벽 규칙은 비활성화합니다.

# systemctl status firewalld
# systemctl stop firewalld

systemctl status firewalld : 방화벽 서비스 설정상태를 확인합니다.

systemctl stop firewalld : 방화벽 서비스가 동작중이라면 방화벽 서비스를 정지합니다.

> 업데이트 관련 설정

# yum remove epel-release -y
# yum update

yum remove epel-release -y : EPEL(Enterprise Linux용 추가 패키지) 최신 저장소 패키지를 삭제하는 명령어입니다. 이 명령어를 실행하면 epel-release 패키지와 관련된 모든 파일과 설정이 시스템에서 제거됩니다. 패키지를 업데이트 하기 전에 다운로드가 시스템의 다른 소프트웨어와 충돌을 일으키는 경우를 대비해서 epel-release를 제거하는 명령어입니다.

yum update : 업데이트를 적용합니다.

# dnf clean all
# rm -r /var/cache/dnf
# dnf upgrade

dnf clean all : 임시 저장 장소의 DNF 관련 캐시된 패키지와 헤더를 삭제합니다.

 DNF(Dandified Yum)는 Yum(Yellowdog Updater Modified)의 후속으로 개발되었으며, Yum의 대안으로서 더 빠르고 효율적인 패키지 관리를 제공합니다.

rm -r /var/cache/dnf : DNF (Dandified YUM) 패키지 매니저의 캐시를 삭제하는 명령어입니다.

dnf upgrade : 현재 설치되어 있는 패키지들을 최신 버전으로 업그레이드합니다.

 

[IPVS 가상 IP(VIP) 설정]
IPVS에서 가상 IP(VIP)는 로드 밸런서의 프런트엔드를 나타내는 고유 IP 주소입니다. 클라이언트는 이 VIP로 요청을 보내면 백엔드 서버에 트래픽을 분산하는 역할을 담당합니다. 

> 네트워크 설정

부팅 시 자동으로 등록되게 VIP 네트워크를 설정합니다.

# vi /etc/sysconfig/network-scripts/ifcfg-ens1:1
DEVICE=ens1:1
BOOTPROTO=static
ONBOOT=yes
IPADDR=192.168.1.1
NETMASK=255.255.255.0

vi /etc/sysconfig/network-scripts/ifcfg-ens1:1 : VIP 네트워크를 설정합니다.

# vi /etc/sysconfig/network-scripts/route-ens1:1
192.168.1.1/32 via 0.0.0.0

vi /etc/sysconfig/network-scripts/route-ens1:1 : VIP 네트워크 라우팅을 설정합니다.

# systemctl restart NetworkManager

systemctl restart NetworkManager : NetworkManager 서비스를 재시작합니다.

> IP 포워딩을 활성화합니다:
로드 밸런서에서 IP 포워딩을 활성화하여 들어오는 트래픽이 백엔드 서버로 올바르게 전달될 수 있도록 합니다. 이는 net.ipv4.ip_forward 커널 파라미터를 1로 설정하면 됩니다.

# vi /etc/sysctl.conf
net.ipv4.ip_forward = 1
# sysctl -p

vi /etc/sysctl.conf : 커널 파라미터 값을 변경하기 설정파일을 수정합니다.

net.ipv4.ip_forward = 1 : net.ipv4.ip_forward의 값을 1로 변경하며 IP 포워딩을 활성화합니다.

sysctl -p : 커널 파라미터 변경사항을 적용합니다.

> ipvsadm 패키지 설정

# yum install ipvsadm
# rm -rf /etc/sysconfig/ipvsadm
# touch /etc/sysconfig/ipvsadm
# systemctl start ipvsadm
# systemctl enable ipvsadm.service
# systemctl status ipvsadm.service
# ipvsadm -C

yum install ipvsadm : 부하분산(Load Balancing)을 구성하기 위한 ipvsadm 패키지를 설치합니다.

rm -rf /etc/sysconfig/ipvsadm : 설정파일을 삭제합니다.

※ ipvsadm 파일에는 IPVS 규칙 및 설정을 구성하는 데 사용되는 변수와 해당 값이 포함되어 있습니다.

touch /etc/sysconfig/ipvsadm : 빈 설정파일을 생성합니다.

systemctl start ipvsadm : ipvsadm 서비스를 시작합니다.

systemctl enable ipvsadm.service : 시스템이 재시작되더라도 자동실행되도록 ipvsadm.service 이름으로 서비스를 등록합니다.

systemctl status ipvsadm.service : ipvsadm.service 이름으로 등록한 서비스의 상태를 확인합니다.

ipvsadm -C : IPVS테이블에 구성된 모든 가상 서버 항목을 초기화합니다.

> IPVS 테이블 구성
로드 밸런싱 규칙 관리를 담당하는 IPVS 테이블을 구성하려면 ipvsadm 명령을 사용합니다. 새 IPVS 테이블을 만들고 각 백엔드 서버에 대한 항목을 추가하여 해당 서버의 IP 주소와 포트, 선택한 부하 분산 스케줄러를 지정합니다.

# ipvsadm -A -t 192.168.1.1:8000 -s lc -p 30
# ipvsadm -a -t 192.168.1.1:8000 -r 192.168.1.10:8000 -g 
# ipvsadm -a -t 192.168.1.1:8000 -r 192.168.1.20:8000 -g 
# ipvsadm --save

ipvsadm -A -t 192.168.1.1:8000 -s lc -p 30 : 서비스로 IPVS VIP를 등록합니다.

ipvsadm -a -t 192.168.1.1:8000 -r 192.168.1.10:8000 -g : 리얼서버 WEB1을 IPVS서비스로 등록합니다.

ipvsadm -a -t 192.168.1.1:8000 -r 192.168.1.20:8000 -g : 리얼서버 WEB2를 IPVS서비스로 등록합니다.

옵션은 아래 따로 설명하겠습니다.

ipvsadm --save : IPVS의 설정을 저장합니다.

> 로드 밸런서 테스트
로드 밸런서를 구성한 후에는 기능과 성능을 철저하게 테스트해야 합니다. VIP에게 테스트 요청을 보내고 선택한 부하 분산 알고리즘에 따라 트래픽이 백엔드 서버 간에 올바르게 분산되는지 확인합니다.

# ipvsadm -Lcn
# ipvsadm -L --stats

ipvsadm -Lcn : 현재 IPVS의 설정을 확인합니다.

 -L : 서비스/리얼서버 리스트를 출력합니다.
 -n : 숫자형식(호스트네임->아이피주소)로 출력합니다.
 -c : 현재 연결을 출력합니다.

ipvsadm -L --stats : 현재 IPVS의 모든 서비스들을 체크합니다.

# cat /etc/sysconfig/ipvsadm
# vi /etc/sysconfig/ipvsadm

cat /etc/sysconfig/ipvsadm : 설정파일 내용을 보여줍니다.

vi /etc/sysconfig/ipvsadm : 설정파일을 수정할 수 있습니다.

# systemctl disable ipvsadm.service
# systemctl start ipvsadm.service
# systemctl stop ipvsadm.service
# systemctl status ipvsadm.service

systemctl disable ipvsadm.service : ipvsadm.service 를 비활성화합니다.

systemctl start ipvsadm.service : ipvsadm.service를 시작합니다.

systemctl stop ipvsadm.service : ipvsadm.service를 정지합니다.

systemctl status ipvsadm.service : ipvsadm.service 이름으로 등록한 서비스의 상태를 확인합니다.

 

[백엔드 서버(WEB1, WEB2) 구성]

# vi /etc/sysctl.conf
net.ipv4.conf.lo.arp_ignore = 1
net.ipv4.conf.lo.arp_announce = 2
net.ipv4.conf.all.arp_ignore = 1
net.ipv4.conf.all.arp_announce = 2
net.ipv4.conf.default.arp_ignore = 1
net.ipv4.conf.default.arp_announce = 2
net.ipv4.conf.ens18.arp_ignore = 1
net.ipv4.conf.ens18.arp_announce = 2

# sysctl -p

vi /etc/sysctl.conf : 커널 파라미터 값을 변경하기 설정파일을 추가합니다.

sysctl -p : 커널 파라미터 변경사항을 적용합니다.


들어오는 요청을 처리할 백엔드 서버를 식별하고 설정 및 작동하는지 확인합니다. 클러스터 전체에서 일관성을 유지하려면 각 백엔드 서버에 동일한 콘텐츠 및 서비스가 있어야 합니다.

 

◎ ipvsadm 관련 상세 내용

[ipvsadm의 기본 문법]

ipvsadm [명령] [프로토콜 옵션] [서비스] [스케줄링 방법] [세션유지 옵션] : 서비스 등록 옵션입니다.
ipvsadm [명령] [프로토콜 옵션] [서비스] -r [리얼서버 옵션] [가중치 옵션] : 리얼서버 등록 옵션입니다.

[서비스/리얼서버 관련 옵션]

> 명령

 -A : 서비스 추가
 -E : 서비스 수정
 -D : 서비스 삭제
 -a : 리얼서버 추가
 -e : 리얼서버 수정
 -d : 리얼서버 삭제
> 프로토콜
 -t : TCP
 -u : UDP
 -f : firewall 서비스 추가(NAT 방식에서 사용)

 

> 기타
 -s [ rr | wrr | lc | wlc | lblc | lblcr | dh | sh | sed | nq ] : 스케줄링 방법 설정

[로드 밸런싱 스케줄링 방식]에서 설명하겠습니다.
 -p [timeout]: 세션 타임아웃 설정
 -r [서버주소:서비스포트] : 리얼서버 설정
 -w [가중치] : 가중치 설정


[설정 확인]
 -L : 서비스/리얼서버 리스트 출력
 -n : 숫자형식(호스트네임->아이피주소)으로 출력
 -c : 현재 연결 출력

[입/출력]
 -S : 표준출력(화면프린트)으로 현재 설정을 내보냅니다.
 -R : 표준입력을 통해 설정을 가져옵니다


[로드 밸런싱 스케줄링 방식]
IPVS는 스케줄러라고도 하는 여러 로드 밸런싱 알고리즘을 지원합니다. 일반적으로 사용되는 알고리즘은 다음과 같습니다:

 

> 라운드 로빈(rr)
round-robin방식은 가장 기본적인 스케줄링 방식으로 "순차적으로" 작업을 각각의 노드에 분배하는 방식입니다. 만약 3 대의 노드(A,B,C)에 round robin 방식을 사용하면, A->B->C->A->B->C와 같이 실행이 됩니다.
> 가중치 라운드 로빈(wrr)
노드별로 가중치를 설정하여 작업을 순차적으로 분배하는 방식으로 여러대의 서버가 각각 성능이 다른경우 성능에 맞춰 가중치를 설정해 주면 성능이 낮은 서버에 과도한 부하가 걸리는 것을 방지할 수 있습니다.
> 최소 연결(lc)
"현재 활성화 되어있는(ESTABLISHED)" 접속량이 적은 서버부터 작업을 할당하는 방식으로, 불필요한 유휴자원이 발생하는 것을 막을 수 있습니다. 만약 접속이 오래 유지되지 않는 경우 이 방식을 사용하면, A->A->A->A->...와 같이 리스트 위쪽에 있는 노드 위주로 작업이 할당됩니다.
> 가중치 최소 연결(wlc)
노드별로 가중치를 설정하여 접속량이 적은 서버부터 작업을 할당하는 방식으로 lc보다 효율적인 자원분배가 가능합니다.

 

[타임아웃]
> 세션 타임아웃
ipvsadm에서 -p 옵션을 이용하여 어떤 서비스에 세션 타임아웃 시간을 설정하면, 한번 세션이 열린후 타임아웃 시간 내에 동일한 주소에서 세션이 다시 열렸을 때 처음에 접속했던 RIP로 접속이 됩니다.
> Idle 타임아웃
ipvsadm --set옵션을 이용하면 프로토콜의 Idle timeout 시간을 설정할 수 있습니다. timeout 시간이 지나면 연결이 끊어지게 됩니다.다. 예를 들어,
ivpsadm --set 10 20 30
로 설정을 해 주게 되고 ftp연결후 10초간 아무것도 하지 않으면 연결이 끊어집니다. 이는 http와 같이 한번 보내면 해당 연결이 끊겨도 별 문제가 없는 서비스의 경우는 상관없지만, ftp, nfs와 같은 서비스를 사용할 때에는 신중하게 값을 설정해야 합니다.

<순서>

[Redhat 계열의 리눅스]

◎ 업그레이드 순서

◎ 커널업데이트 없이 업그레이드

◎ 특정버전 커널업그레이드(자동)

◎ 특정버전 커널업그레이드(수동)

◎ 구커널이미지 삭제



<내용>

[Redhat 계열의 리눅스]

OS종류 : RHEL, CentOS, Fedora, SUSE 등

패키지 설치, 업데이트, 삭제 명령어 : rpm(Redhat Package Manager), yum(Yellowdog Updater Modified)


◎ 업그레이드 순서

1. 패키지 정보 업데이트

 # yum -y update

각 업데이트 저장소에서 업데이트 패키지 목록을 갱신합니다.


2. 패키지 업그레이드

 # yum upgrade

패키지 목록과 보유한 목록을 비교하여 실제 패키지 업데이트를 수행합니다.


◎ 커널업데이트 없이 업그레이드

1. 커널업데이트 금지 설정

 # yum update --exclude=kernel*

커널업데이트를 제외합니다. 또는 /etc/yum.conf에서 다음 문장을 추가해줍니다. exclude=kernel*


2. 패키지 업그레이드

 # yum upgrade

패키지 목록과 보유한 목록을 비교하여 실제 패키지 업데이트를 수행합니다.


◎ 특정버전 커널업그레이드(자동)

1. 커널 정보 확인

 # uname -r

또는

 # cat /proc/version 

현재 부팅되어있는 커널 정보를 확인합니다.


2. 리눅스 커널 이미지 버전 검색

 http://vault.centos.org/

http://vault.centos.org/ 에서 리눅스 커널 이미지 버전을 검색합니다


3. 패키지 저장소(Repository) URL 수정

 # cd /etc/yum.repos.d

 # cp CentOS-Base.repo CentOS-Base.repo.bak

 # vi CentOS-Base.repo

 [base]

 baseurl=http://vault.centos.org/버전/os/x86_64/

 gpgkey=http://vault.centos.org/버전/os/x86_64/RPM-GPG-KEY-CentOS-버전

 [updates]

 baseurl=http://vault.centos.org/버전/updates/x86_64/

 gpgkey=http://vault.centos.org/버전/os/x86_64/RPM-GPG-KEY-CentOS-버전

 [extras]

 baseurl=http://vault.centos.org/버전/extras/x86_64/

 gpgkey=http://vault.centos.org/버전/os/x86_64/RPM-GPG-KEY-CentOS-버전

 [centosplus]

 baseurl=http://vault.centos.org/버전/centosplus/x86_64/

 gpgkey=http://vault.centos.org/버전/os/x86_64/RPM-GPG-KEY-CentOS-버전

 [contrib]

 baseurl=http://vault.centos.org/버전/contrib/x86_64/

 gpgkey=http://vault.centos.org/버전/os/x86_64/RPM-GPG-KEY-CentOS-버전

편집을 통해 mirrorlist 대신 baseurl 을 위 내용으로 추가하여 대신하고 gpgkey를 수정합니다.


4. 패키지 정보 업데이트

 # yum -y update

각 업데이트 저장소에서 업데이트 패키지 목록을 갱신합니다.


5. 패키지 업그레이드

 # yum upgrade

패키지 목록과 보유한 목록을 비교하여 실제 패키지 업데이트를 수행합니다.


6. 재부팅

 # reboot


◎ 특정버전 커널업그레이드(수동)

1. 커널 다운로드

 http://www.kernel.org

또는

 # wget https://cdn.kernel.org/pub/linux/kernel/v3.x/linux-버전.tar.zx

또는

 # wget https://cdn.kernel.org/pub/linux/kernel/v3.x/linux-버전.tar.gz

www.kernel.org 에서 원하는 커널버전의 tar.xz 를 다운로드 받습니다.

또는 링크주소를 복사하여 wget으로 tar.xz 이나 tar.gz 를 다운로드 받습니다.


2. 다운로드 파일 이동

 # mv linux-버전.tar.zx /usr/src

또는

 # mv linux-버전.tar.gz /usr/src

다운로드 받은 파일을 /usr/src 로 이동합니다.


3. 압축해제

 # xz -d linux-버전.tar.zx

 # tar -xvf linux-버전.tar

tar 와 xz 포맷으로 압축된 파일의 압축을 해제합니다.

또는

 # tar -xvzf linux-버전.tar.gz

tar.gz 포맷으로 압축된 파일의 압축을 해제합니다.


4. 심볼릭링크

 # ln -s /usr/src/linux-버전 /usr/src/linux


5. 환경설정

 # cd /usr/src/linux

 # make mrproper

 # make clean

make mrproper : 기존에 설정되어 있던 모든 의존관계 및 환경 설정  값들을 제거 합니다

make clean : 기존 소스 컴파일 작업으로 생성된 파일들 삭제합니다.


6. 컴파일할 소스파일 복사

 # cp /boot/config-`uname-r` ./.config

현재 사용하고 있는 커널 설정 파일을 복사합니다.


7. 커널환경설정

 # make menuconfig

Load 메뉴를눌러 .config 소스 파일을 불러옵니다. 소스파일 로딩이 완료되면 General setup 항목을 <Enter> 를 이용해 선택합니다.

Enable deprecated sysfs features to support old userspace too 라는 항목을 <Space bar> 를 이용해 체크하고 저장합니다.


::추가사항::

화면이 깨진다면 # yum install gcc* ncurses-devel 을 실행 후 진행하시기 바랍니다.

gcc : 컴파일에 필요한 gcc 컴파일러 설치

ncurses-devel : 텍스트 기반 화면처리 라이브러리


8. 커널 컴파일

 # make all

커널 설정 파일 .config를 사용하여  커널 소스를 컴파일 합니다


9. 커널 설치

 # make modules_install

 # make install

컴파일된 모듈들을 설치합니다.


10. 커널 적용을 위한 default 부팅 값 수정

 # vi /boot/grub/grub.conf

새로 설치한 커널이 첫번째 라인에 있으므로 default=0 으로 설정합니다.


◎ 구커널이미지 삭제

리눅스는 커널 업데이트가 빈번합니다. 그리고 매번 업그레이드할 때마다, 이전의 커널 이미지는 삭제되지 않고 남아 있습니다. 때문에 커널 이미지를 삭제해주는 작업이 필요합니다.

1. 커널 정보 확인

 # uname -r

또는

 # cat /proc/version 

현재 부팅되어있는 커널 정보를 확인합니다.


2. 커널 리스트 확인

 # rpm -q kernel

현재 설치되어있는 커널 리스트를 확인합니다.


3.1. 커널 이미지 삭제

 # yum erase kernel-버전

불필요한 커널을 삭제합니다.


종합

 # yum install yum-utils

 # package-cleanup --oldkernels --count=1

yum-list를 이용해서 위의 1~3 단계를 한번에 실행합니다. count 는 구버전의 커널을 몇개까지 남겨 놓을지 설정하는 옵션입니다. 기본 값은 5입니다.



<순서>

[Debian 계열의 리눅스]

◎ 업그레이드 순서

◎ 커널업데이트 없이 업그레이드

◎ 특정버전 커널업그레이드(자동)

◎ 특정버전 커널업그레이드(수동)

◎ 구커널이미지 삭제



<내용>

[Debian 계열의 리눅스]

OS종류 : Debian, Ubuntu, Kali 등

패키지 설치, 업데이트, 삭제 명령어 : dpkg(package manager for Debian), apt-get(Advanced Packaging Tool package handling utility)


업그레이드 순서

1. 패키지 정보 업데이트

 # apt-get update

실제 패키지를 설치하지 않고 신규버전 패키지의 정보를 다운받아 이용 가능한 패키지의 정보를 /etc/apt/sources.list 파일에 업데이트 패키지 목록을 갱신하며 update 는 항상 upgrade 전에 실행되어야 합니다.


2.1. 패키지 업그레이드

 # apt-get upgrade -y

update 로부터 패키지 목록과 보유한 목록을 비교하여 장비내에 존재하는 실제 패키지 업그레이드를 수행합니다.

가장최신버전으로 설치하기 위해 사용되며 설치된 패키지가 제거되거나 설치되어 있지 않은 패키지가 설치되는 작업은 하지 않습니다.

-y : 프롬프트에 자동으로 yes라는 메시지가 표시됩니다. 모든 프롬프트에 대한 응답을 "예"라고 가정하고 비대화식으로 실행합니다.


2.2. 패키지 업그레이드 및 추가 패키지 설치

 # apt-get dist-upgrade

의존성 검사를 수행하면서 새로운 버전으로 업그레이드를 수행합니다.

컴퓨터에 설치된 꾸러미는 아니지만 사용에 필요한 추가 꾸러미를 설치하는 작업도 수행하기 때문에 이미 설치된 꾸러미를 최신 판으로 바꿔주는 upgrade 와는 차이가 있습니다.


◎ 커널업데이트 없이 업그레이드

1. 커널업데이트 금지 설정

 # apt-mark hold linux-image-generic linux-headers-generic

커널업데이트를 금지합니다.


2. 패키지 정보 업데이트

 # apt-get update


3. 패키지 업그레이드

 # apt-get upgrade -y


종합

 # apt-mark hold linux-image-generic linux-headers-generic && apt-get update && sleep 1 && apt-get upgrade -y

위의 1~3 단계를 한번에 실행합니다.


특정버전 커널업그레이드(자동)

1. 커널 정보 확인

 # uname -r

또는

 # cat /proc/version 

현재 부팅되어있는 커널 정보를 확인합니다.


2. 패키지 정보 업데이트

 # apt-get update


3. 리눅스 커널 이미지 버전 검색

 # apt-cache search linux-image

패키지 관리자의 리스트에서 리눅스 커널 이미지 버전을 검색합니다


4. 리눅스 커널 이미지 설치

 # apt-get install linux-image-버전

검색한 여러 버전중 가장 최신 버전을 위 명령어를 통해 입력하면 해당 커널 버전에 관련된 커널 패키지가 같이 설치됩니다.


5. 재부팅

 # reboot


◎ 특정버전 커널업그레이드(수동)

1. 커널 정보 확인

 # uname -r

또는

 # cat /proc/version 

현재 부팅되어있는 커널 정보를 확인합니다.


2. 패키지 다운로드

 # wget http://kernel.ubuntu.com/~kernel-ppa/mainline/해당내용

먼저 http://kernel.ubuntu.com/~kernel-ppa/mainline 에 접속해서 확인 후 필요 버전을 다운로드 합니다.


3. 패키지 설치

 # dpkg -i linux-headers-버전*.deb linux-image-버전*.deb

또는

 # dpkg -i *.deb 


4. grub2 업데이트

 # update-grub2

커널 이미지 정보가 반영되도록 grub2를 업데이트합니다.


5. 재부팅

 # reboot 


구커널이미지 삭제

리눅스는 커널 업데이트가 빈번합니다. 그리고 매번 업그레이드할 때마다, 이전의 커널 이미지는 삭제되지 않고 남아 있습니다. 때문에 커널 이미지를 삭제해주는 작업이 필요합니다.


1. 커널 정보 확인

 # uname -r

또는

 # cat /proc/version 

현재 부팅되어있는 커널 정보를 확인합니다.


2. 커널 리스트 확인

 # dpkg --list | grep linux-image

현재 설치되어있는 커널 리스트를 확인합니다.


3.1. 커널 이미지 삭제

 # apt-get remove linux-image-버전-generic

불필요한 커널 이미지를 삭제합니다.


3.2. 커널 이미지 삭제(설정파일 포함)

 # apt-get --purge remove linux-image-버전-generic

불필요한 커널 이미지와 함께 설정파일까지 모두 삭제합니다.


종합

 # purge-old-kernels --keep 1

purge-old-kernels 를 이용해서 위의 1~3 단계를 한번에 실행합니다. keep 는 구버전의 커널을 몇개까지 남겨 놓을지 설정하는 옵션입니다. 기본 값은 2입니다.


4. grub2 업데이트

 # update-grub2

삭제된 커널 이미지 정보가 반영되도록 grub2를 업데이트합니다.


5. 재부팅

 # reboot 


ELF statifier 는 동적 링크된 실행 파일과 공유 라이브러리를 하나의 파일로 합치기 위한 툴로 동적 실행 파일에서 정적 실행 파일을 생성합니다. 침해사고 시스템을 분석할 때는 라이브러리에 의존하지 않기 위해 정적 실행 파일을 사용해야 하지만 정적 컴파일 된 파일은 사이즈가 크다는 단점이 존재합니다.

 

동적으로 링크 된 공유 라이브러리는 정적으로 링크 된 공유 라이브러리보다 디스크 공간을 효율적으로 사용하며, 보다 효율적인 방식으로 보안 업데이트를 수행 할 수 있습니다. 그리고 동적 라이브러리의 특정 버전으로 컴파일 된 실행 파일은 해당 버전의 공유 라이브러리가 실행중인 시스템에서만 사용이 가능합니다.

 

Fedora 9 와 openSUSE 11 를 각각 실행해 보면 일부 공유 라이브러리의 버전이 다른 것을 볼 수 있으며 이러한 버전 차이로 인해 컴퓨터간에 실행 파일을 복사하더라도 파일을 실행할 수 없습니다. 하지만 ELF Statifier 를 이용하면 동적 라이브러리를 찾는 실행 파일 대신에 공유 라이브러리를 포함한 정적으로 링크 된 실행 파일을 만들 수 있으며 Fedora 컴퓨터에서 컴파일 한 실행 파일을 다시 컴파일하지 않고도 openSUSE 시스템에서 실행할 수 있습니다. 물론 실행 파일을 복사하는 시스템과 실행 파일을 복사하려는 시스템은 동일한 아키텍처를 가져야합니다.

 

물론 이렇게하려면 정적 실행 파일에 필요한 공유 라이브러리 복사본이 포함되어 있기 때문에 일부 디스크 공간을 희생해야하지만 요즘에는 디스크 공간이 크기 때문에 중요하지 않습니다. 실행 파일이 libfoo 라는 공유 라이브러리에 동적으로 링크되어 있고 libfoo 에 대한 보안 업데이트가 있는 경우 어떻게 될지 고려해 보십시오. 응용 프로그램이 동적으로 링크되어 있다면 libfoo 의 공유 사본을 업데이트 할 수 있으며 응용 프로그램은 더 이상 이전 libfoo 의 보안 문제로 취약하지 않습니다. 반면에 정적으로 링크 된 실행 파일의 경우 이전 libfoo 라는 공유 라이브러리 복사본이 포함되어 있기 때문에 최신 libfoo 및 보안 업데이트를 적용하려면 정적으로 링크 된 실행 파일을 다시 만들어야 합니다.

 

[Statifier 설치]

ELF Statifier 는 openSUSE 10.3 에서는 1-Click 으로 설치되지만 Ubuntu 나 Fedora 에서는 1-Click 으로 설치되지 않습니다. 여기서는 ELF Statifier 1.6.14 버전을 사용했으며 ELF Statifier 는 자동 도구를 사용하지 않으므로 간단하게 호출하여 컴파일 할 수 있습니다. make 컴파일과 설치는 아래와 같습니다.

 

 # tar xzvf statifier-1.6.14.tar.gz 
 # cd ./statifier-* 
 # make 
 # make install

 

[ASLR 해제]

정적 실행 파일을 만들기 전 메모리상의 공격을 방어하기 위해 주소 공간배치를 난수화 시키는 기법(ASLR : Address Space Layout Randomization) 은 비활성화해야 합니다. 주소 공간배치를 난수화하면 실행 파일을 실행할 때마다 libc 함수의 위치가 항상 바뀌기 때문에 return-to-libc 공격과 같은 BOF 공격으로부터 메모리를 보호하게 됩니다. 하지만 정상적인 정적 실행 파일 또한 실행이 불가합니다.

([Linux] 메모리 보호 기법[Linux] 메모리 보호 기법 중 ASLR와 NX 설정값 확인 및 변경글을 참고해주세요.)

 

아래와 같이 randomize_va_space 매개 변수를 변경하여 보안을 해제합니다. randomize_va_space 변경은 실행 파일 자체에 대한 공격뿐만 아니라 버퍼 오버 플로우에 의존하여 시스템을 손상시키는 공격 시도에도 영향을 미칩니다. 주소 공간배치 난수화가 없으면 두 공격에 더 취약 해집니다.

아래와 같이 randomize_va_space를 0으로 설정하고 정적 실행 파일을 만들면 됩니다.

 # cd /proc/sys/kernel
 # cat randomize_va_space 
 2 
 # echo 0 > randomize_va_space 
 # cat randomize_va_space 
 0

 

[정적 실행 파일 생성]

다음으로 statifier 유틸리티를 사용하는 방법의 예로서, ls 명령에 대한 정적 실행 파일을 생성해보겠습니다.

먼저 동적으로 링크 된 실행 파일의 복사본을 만들고 동적으로 링크 된 파일을 검사합니다.

statifier 는 동적으로 링크 된 실행 파일의 경로를 첫 번째 인수로 사용하고 정적으로 링크 된 실행 파일을 두 번째 인수로 만들려는 경로로 실행합니다.

ldd 명령은 ls-static 이 동적으로 링크 된 라이브러리를 필요로하지 않다고 보여주고 있습니다.

다음 명령은 정적 실행 파일 ls 에서 바이너리 크기가 크게 증가한 것을 보여줍니다.

 

 # mkdir test
 # cd ./test 
 # cp -a /bin/ls ls-dynamic 
 # ls -lh 
 -rwxr-xr-x 1 root root 112K 2008-08-01 04:05 ls-dynamic 
 # ldd ls-dynamic 
 linux -gate.so.1 => (0x00110000) 
 librt.so.1 => /lib/librt.so.1 (0x00a3a000) 
 libselinux.so.1 => /lib/libselinux.so.1 (0x00a06000) 
 libacl.so .1 => /lib/libacl.so.1 (0x00d8a000) 
 libc.so.6 => /lib/libc.so.6 (0x0084e000) 
 libpthread.so.0 => /lib/libpthread.so.0 (0x009eb000 ) 
 /lib/ld-linux.so.2 (0x0082e000) 
 libdl.so.2 => /lib/libdl.so.2 (0x009e4000) 
 libattr.so.1 => /lib/libattr.so.1 (0x0606d000)
 # statifier ls-dynamic ls-static 
 # ldd ls-static 
 not a dynamic executable
 # ls -lh ls-static 
 -rwxr-x --- 1 root root 2.0M 2008-10-03 12:05 ls-static
 # ls-static / tmp 
 ... 

 

[문제해결1]

 # ls-static -lh 
 Segmentation fault

statified 실행 파일을 실행할 때 세그먼테이션 오류가 발생하면 스택 무작위화를 비활성화하고 statified 실행 파일을 다시 작성해야 합니다. 또한 정적 컴파일된 파일을 복사 후 실행하려면 ASLR 기능을 사용하지 않아야 합니다.

 

[문제해결2]

 조건  A-PC CentOS 실행 A-PC Ubuntu 실행 B-PC CentOS 실행 B-PC Ubuntu 실행
 A-PC CentOS에서
컴파일
 O   O   X  X
 B-PC CentOS에서
컴파일
 O  O  O  O

PC가 달라지면 실행이 안되는 경우도 발생 할 수 있으니 충분히 테스트해보시기 바랍니다.

 

[참고사이트]

원문 : https://www.linux.com/news/quickly-move-executable-between-systems-elf-statifier

ELF statifier 설명 : http://statifier.sourceforge.net/

ELF statifier 다운로드 : https://sourceforge.net/projects/statifier/

1. ASLR(Address Space Layout Randomization) 설정 값 확인 및 변경

ASLR(Address Space Layout Randomization) 설정 값을 확인합니다.

 # cat /proc/sys/kernel/randomize_va_space

 

randomize_va_space=0 //ASLR 해제

randomize_va_space=1 //랜덤 스택 & 라이브러리 활성화

randomize_va_space=2 //랜덤 스택 & 라이브러리 & 힙 활성화

 

ASLR(Address Space Layout Randomization) 설정 값을 변경합니다. 임시로 변경하는 방법이기 때문에 재부팅되면 설정 값이 기본 값(2)으로 변경됩니다.

 # echo 0 > /proc/sys/kernel/randomize_va_space
 # echo 1 > /proc/sys/kernel/randomize_va_space
 # echo 2 > /proc/sys/kernel/randomize_va_space

 

2. DEP / NX(Not Excutable) 설정 값 확인 및 변경

DEP / NX(Not Excutable) 설정 값을 확인합니다. 

 # cat /proc/sys/kernel/exec-shield

 

exec-shield=0 // ExecShield 보호 비활성화

exec-shield=1 // ExecShield 보호 활성화

 

DEP / NX(Not Excutable) 설정 값을 변경합니다. 임시로 변경하는 방법이기 때문에 재부팅되면 설정 값이 기본 값(1)으로 변경됩니다.

# echo 0 > /proc/sys/kernel/exec-shield
 # echo 1 > /proc/sys/kernel/exec-shield

 

3. sysctl 명령을 이용한 설정값 확인 및 변경

sysctl 명령은 커널 변수의 값을 제어하여 시스템을 최적화 할 수 있는 명령으로 시스템의 /proc/sys 디렉토리 밑에 있는 매개변수를 제어합니다. echo 명령을 사용하거나 sysctl 명령을 이용하여 /proc 디렉토리 밑에 있는 항목을 직접 편집하거나 설정할 수 있습니다.

 

sysctl 옵션
 -a : 현재 커널 매개변수와 값을 보여줍니다
 -p : sysctl 환경변수파일(기본값은 /etc/sysctl.conf)의 설정상태를 보여줍니다
 -n : 특정키에 대한 값을 보여줍니다.
 -w variable=value : 변수에 값을 설정합니다.
 -A : 테이블형재로 설정가능한 파라미터를 보여줍니다. -a 와 같습니다.

 

sysctl 명령을 이용하여 ASLR(Address Space Layout Randomization) 설정 값과 DEP / NX(Not Excutable) 설정 값을 확인합니다.

# sysctl -a | grep -E "exec-|randomi"
 kernel.exec-shield=1
 kernel.randomize_va_space=2

 

sysctl 명령을 이용하여 ASLR(Address Space Layout Randomization) 설정 값을 변경합니다. 임시로 변경하는 방법이기 때문에 재부팅되면 설정 값이 기본 값(2)으로 변경됩니다.

# sysctl -w kernel.randomize_va_space=0
 # sysctl -w kernel.randomize_va_space=1
 # sysctl -w kernel.randomize_va_space=2

 

sysctl 명령을 이용하여 DEP / NX(Not Excutable) 설정 값을 변경합니다. 임시로 변경하는 방법이기 때문에 재부팅되면 설정 값이 기본 값(1)으로 변경됩니다.

# sysctl -w kernel.exec-shield=0
 # sysctl -w kernel.exec-shield=1

 

시스템을 시작할 때부터 ASLR(Address Space Layout Randomization) 설정과 DEP / NX(Not Excutable) 설정 해제를 자동으로 설정하는 방법

/etc/sysctl.conf 파일 안에 아래 내용을 추가한 후 재부팅을 합니다.

# vi /etc/sysctl.conf
 kernel.randomize_va_space=0
 kernel.exec-shield=0

 

다른 메모리 보호 기법은 [Linux] 메모리 보호 기법을 참고해주세요.

[리눅스 메모리 보호 기법]

1. ASLR(Address Space Layout Randomization)

2. DEP / NX(Not Excutable)

3. ASCII-Armor

4. Stack canary

 

예제)

 # cat /proc/self/maps

스택, 힙, 라이브러리 등의 주소가 랜덤하게 바뀌는지 확인이 가능합니다.

 

1. ASLR(Address Space Layout Randomization)

 

메모리상의 공격을 방어하기 위해 주소 공간배치를 난수화 시키는 기법입니다. 스택, 힙, 라이브러리 등의 데이터 영역 주소등을 난수화 시킨 주소로 프로세스의 주소 공간에 배치하는 것입니다. 리눅스 커널 2.6.12 이후로 적용 되었습니다.

(상세한 내용은 [Linux] 메모리 보호 기법 중 ASLR와 NX 설정값 확인 및 변경을 참고해주세요)

 

2. DEP / NX(Not Excutable)

 

메모리상의 보호를 위해 stack 과 heap 에서 코드가 실행되는 것을 막는 기법입니다. 공격자가 BOF 공격을 일으키면 DEP 가 적용된 상태에서는 실행권한이 없으므로 프로그램에 대한 예외처리 후 종료가 됩니다.

 

3. ASCII-Armor

 

Libc 영역을 보호하기 위한 기법으로 상위주소를 \x00 으로 시작하게 만들어 공격자가 라이브러리를 호출하는 BOF 공격을 하여도 NULL 바이트가 삽입된 주소로 접근 할 수 없게 됩니다.

 

4. Stack canary

 

RET
SFP
BUFFER

 

위 처럼 되있는 메모리 구조가

 

RET
SFP
CANARY
BUFFER

 

이렇게 SFP와 BUFFER 데이터 사이에 CANARY 가 추가되어 스택의 BOF를 모니터링하는 역할을 담당합니다. BOF 가 발생하면 CANARY 데이터값의 변조로 오버플로우에 대한 경고를 하고 프로그램을 종료시킵니다. 

 

CANARY 데이터는 다음과 같이 구성되어 있습니다.

 

 1) Terminator canaries
 문자열 끝문자를 이용하여 canary 를 구성하는 방법으로 canary 값으로 NULL, CR, LF, Oxff 값의 조합이 사용됩니다. 공격자는 공격시,  종료문자로 구성된 canary 값에 접근을 할 수 없게됩니다.


 2) Random canary
 프로그램을 실행할 때마다 임의의 canary 값을 삽입을 합니다. 이에 따라 공격자는 Canary 값을 예측불가능하므로 공격에 어려움이  발생합니다.


 3) Null canary(0x00000000)
 메모리상의 공격을 막기위해 canary 값을 NULL 로 구성합니다. 공격자는 공격코드상에 NULL 값을 삽입할 수 없으므로 canary 값에  접근이 불가능합니다.

 

[참고 사이트]

https://bpsecblog.wordpress.com/2016/05/16/memory_protect_linux_1/

http://dokydoky.tistory.com/449

http://blog.naver.com/revsic/220531540037

+ Recent posts