Rockylinux8 에서의 PHP 8.3 컴파일 시 나올 수 있는 에러와 이에 대한 대응방법 정리.
1. mbstring 관련
/usr/bin/ld: dynamic STT_GNU_IFUNC symbol `mb_utf16be_to_wchar' with pointer
equality in `ext/mbstring/libmbfl/filters/mbfilter_utf16.o' can not be used
when making an executable; recompile with -fPIE and relink with -pie
collect2: error: ld returned 1 exit status
make: *** [Makefile:307: sapi/cli/php] Error 1
아래와 같은 오류가 발생하면, 99.9% /etc/mail/local-host-names 설정이 잘못되어 있기 때문이다.
----- The following addresses had permanent fatal errors -----
<center@center.co.kr>
(reason: 553 5.3.5 system config error)
----- Transcript of session follows -----
553 5.3.5 217.129.242.1. config error: mail loops back to me (MX problem?)
554 5.3.5 Local configuration error
local-host-names 파일에 center.co.kr 도메인 이름을 추가하고,
sendmail 데몬을 다시 시작하면, 메일을 받을 수 있다.
물론, center.co.kr 도메인에 대한 MX 레코드가 해당 메일 서버로 지정되어 있어야만 한다.
제 목 : find명령에서 특정 디렉토리 제외하고 찾기(exclude) 작성자 : 좋은진호(truefeel, http://coffeenix.net/ ) 작성일 : 2009.7.3(금)
find 명령을 할 때, 특정 디렉토리 제외하고 find하는 방법은 없느냐고 주위분이 물어오셨다. 개인적으로는 find에서 특정 디렉토리를 exclude하는 것 대신에 디렉토리를 나열하는 방법을 사용했다. 그러나 디렉토리 갯수가 많고, 같은 depth의 디렉토리가 아니고 하위 특정 디렉토리 몇개를 exclude해야한다면? 옵션을 활용할 수 밖에 없을 것이다.
man페이지를 확인해보니 -prune옵션이 있다. -prune 옵션은 찾아낸 것이 디렉토리이면 그 디렉토리내에는 find하지 않는다.
-name pattern ... 생략 ... To ignore a directory and the files under it, use -prune; see an example in the description of -wholename.
-prune If -depth is not given, true; if the file is a directory, do not descend into it. If -depth is given, false; no effect.
1. 예제로 살펴보기
[예제 1] 2009가 포함된 파일을 찾는다. 단 파일명이 디렉토리에 해당되면 해당 디렉토리 이하는 찾지 않는다.
-path pattern See -wholename. The predicate -path is also supported by HP-UX find.
-wholename pattern File name matches shell pattern pattern. The metacharacters do not treat `/' or `.' specially; so, for example, find . -wholename './sr*sc' will print an entry for a directory called './src/misc' (if one exists). To ignore a whole directory tree, use -prune rather than checking every file in the tree. For example, to skip the directory `src/emacs' and all files and directories under it, and print the names of the other files found, do something like this: find . -wholename './src/emacs' -prune -o -print
DNS 관련 포트 설정시TCP 53번, UDP 53번을 허용해야 한다. (다시 말해 네임서버 소프트웨어는 TCP도 지원을 할 필요가 있다) [RFC1123 - 6.1.3.2] DNS resolvers and recursive servers MUST support UDP, and SHOULD support TCP, for sending (non-zone-transfer) queries. https://tools.ietf.org/html/rfc1123#section-6.1.3.2
의무 사항은 아니나 권장이 되는 사항이다.
일반적인 DNS 질의는 UDP를 통해 이루어진다. DNS 질의과정이 TCP를 쓰는 경우는 다음과 같다.
1. Zone-Transfer(AXFR/IXFR)
Master-Slave 구성시 Zone Transfer가 이루어지는데, 이때는TCP를 사용하여 Zone File을 주고받는다. 전체 복사인 AXFR, 증분 복사인 IXFR 모두 TCP를 사용한다.
(BIND9 Zone-Transfer 로그)
2. 메시지가 512byte를 초과하는 경우
DNS 메시지에는 UDP 사용시 512byte 제한이 있다.
일반적인 DNS 메시지가 512byte를 초과하는 경우는 거의 없으나, DNSSEC, IPv6 등에서 512byte를 초과하는 일이 종종 발생한다.
특히, NSEC3 레코드 등은 거의 필수적으로 512byte를 초과한다.
서버에서 주려는 응답이 512byte를 초과하는 경우, TC Flag를 이용하여 DNS 질의자와 응답자는TCP를 통해 질의 응답을 진행하게 된다.
(NSEC3 레코드는 사실상 512바이트를 초과할 수 밖에 없다)
요즘은EDNS0 메커니즘에 의해, EDNS0을 지원하는 조합이라면 512byte 이상의 UDP 메시지를 보낼 수 있다. (권장되는 EDNS Payload는 512~4096byte이다)
하지만,모든 클라이언트가 EDNS0을 지원하는 것은 아니며, 일부 오래된 방화벽이나 네트워크 정책 등에 의하여 512byte가 넘는 UDP 질의응답이 차단될 수도 있다.
리눅스 서버는/etc/machine-id라는 파일이 있다. 해당 파일은 machine의 고유 아이디값을 랜덤하게 생성하여 저장된 것으로 32글자로 된 16바이트/128비트 의 UUID값이다.
이machine-id는 dbus 또는 systemd 동작에 관련되어 있으며/etc/machine-id는 기본적으로/var/lib/dbus/machine-id에 심볼릭 링크되어있는 경우도 있다.
일단 해당 파일이 없는경우 어떤 문제가 일어나는지 알아보자
[ 2.680252] systemd[1]: Inserted module 'ip_tables'
[ 3.751916] systemd-journald[665]: Failed to get machine id: No such file or directory
[ 3.755044] systemd-journald[677]: Failed to get machine id: No such file or directory
[ 3.757980] systemd-journald[678]: Failed to get machine id: No such file or directory
[ 3.760862] systemd-journald[679]: Failed to get machine id: No such file or directory
[ 3.763454] systemd-journald[680]: Failed to get machine id: No such file or directory
[ 4.023763] systemd-tmpfiles[676]: [/usr/lib/tmpfiles.d/systemd.conf:26] Failed to replace specifiers: /run/log/journal/%m
[ 4.023804] systemd-tmpfiles[676]: [/usr/lib/tmpfiles.d/systemd.conf:28] Failed to replace specifiers: /run/log/journal/%m
[ 4.023838] systemd-tmpfiles[676]: [/usr/lib/tmpfiles.d/systemd.conf:29] Failed to replace specifiers: /run/log/journal/%m
[ 4.023873] systemd-tmpfiles[676]: [/usr/lib/tmpfiles.d/systemd.conf:32] Failed to replace specifiers: /var/log/journal/%m
[ 4.023910] systemd-tmpfiles[676]: [/usr/lib/tmpfiles.d/systemd.conf:33] Failed to replace specifiers: /var/log/journal/%m/system.journal
[ 4.031224] systemd-tmpfiles[676]: [/usr/lib/tmpfiles.d/systemd.conf:37] Failed to replace specifiers: /var/log/journal/%m
[ 4.031259] systemd-tmpfiles[676]: [/usr/lib/tmpfiles.d/systemd.conf:38] Failed to replace specifiers: /var/log/journal/%m
[ 4.031292] systemd-tmpfiles[676]: [/usr/lib/tmpfiles.d/systemd.conf:39] Failed to replace specifiers: /var/log/journal/%m/system.journal
시스템의dmesg내용을 보면 위와같은 항목이 나타나게 된다. 이것은 systemd가 부팅시machine-id를 읽어 관련 동작을 초기화하는데 있어machine-id값이 없어서 나는 오류메시지 이다. 파일이 없으면 자동으로 생성하면 되는게 아닌가 싶은데 일단 파일이 없는경우 자동생성되지 않고 위와같이 오류메세지가 나타난다.
또한 이 오류로 인해 systemd 관련 로그가 생성되지 않는다. 즉 대부분의/var/log/에 쌓이는 로그들이 정상적으로 기록되지 않는다.
해결법
위 증세를 해결하기 위해서는 반드시/etc/machine-id라는 파일이 존재하여야 한다. 두가지 방법이 있는데 아래를 참조하면 된다.
서버에서 파일명이 깨지는 경우 인코딩 변경으로 문제를 해결할 수 있는 명령어이다. 이종간의 OS (예 : 윈도우와 MAC) 에서 주로 발생하는데 윈도우는 EUC-KR이 기본 인코딩이고 MAC은 UTF-8이 기본 인코딩이라 MAC에서 FTP등을 통해 한글파일이나 폴더를 업로드할 경우 인코딩이 깨진채로 업로드된다. 이런 경우 convmv라는 유용한 명령어가 있는데 명령어가 없다면 아래 명령어로 설치한다. (CentOS 또는 RockyLinux) yum install convmv
폴더로 들어가서 처리를 할 경우 아래와 같이 명령어를 선택하면 해당 폴더에 있는 euc-kr 인코딩이 utf-8 인코딩 방식으로 바뀐다.
[EUC-KR → UTF-8]
convmv -f euc-kr -t utf-8 ./*.* --notest
[UTF-8 → EUC-KR]
convmv -f utf-8 -t euc-kr ./*.* --notest
여기서 --notest 를 입력하지 않을 경우 인코딩 방식은 바뀌지 않고, 어떻게 변경되는지만 확인할 수 있다.
참고로 현재 위치에서 하위 디렉토리까지 포함하여 인코딩 방식을 변경하기 위해서는 -r 옵션을 추가하면 된다.