Microsoft MVP성태의 닷넷 이야기
오류 유형: 773. shell script 실행 시 "$'\r': command not found" 오류 [링크 복사], [링크+제목 복사],
조회: 13881
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

shell script 실행 시 "$'\r': command not found" 오류

예를 들어 볼까요? ^^ 윈도우 환경에서 다음과 같이 간단한 shell script를 만들고,

echo "TEST"

이를 WSL 환경, 또는 리눅스 환경에서 실행하면 다음과 같이 오류가 발생합니다.

$ ./test.sh
./test.sh: line 1: $'\r': command not found
TEST
./test.sh: line 3: $'\r': command not found
./test.sh: line 5: syntax error near unexpected token `$'{\r''
'/test.sh: line 5: `{

왜냐하면, 리눅스의 경우 개행 처리가 0x0a('\n') 단일 문자이지만, 윈도우의 경우는 0x0d('\r'), 0x0a('\n') 2개의 문자이기 때문입니다. 게다가 리눅스의 bash 쉘은 '\r' 문자를 파싱해야 할 대상으로 취급하니 문제가 됩니다.




재미있는 것은, 저 규칙이 git으로 관리하는 프로젝트의 경우 혼란스러운 상황을 연출할 수 있습니다.

가령, 윈도우에서 개발하느라 리눅스에서의 사용을 고려해 ".sh" 파일만 특별하게 '\n' 단일 개행을 해서 저장했는데, 이 파일을 다른 윈도우 환경에서 git fetch로 업데이트를 받는 경우 '\r', '\n'으로 자동 변경이 됩니다.

이 규칙은 git의 core.eol / core.autocrlf로 인한 것인데요,

git 에서 CRLF 개행 문자 차이로 인한 문제 해결하기
; https://www.lesstif.com/gitbook/git-crlf-20776404.html

따라서, git fetch로 인한 업데이트 시에 ".sh" 파일의 개행을 윈도우 환경에서도 그대로 받으려면 "core.autocrlf"를 false로 설정해야 합니다.

그런데, 위의 글에 보면 false가 기본 설정이라고 하는데 비주얼 스튜디오와 함께 설치되는 git 클라이언트의 경우에는 기본값이 true로 설정돼 있습니다.

C:\temp> where git
C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\Common7\IDE\CommonExtensions\Microsoft\TeamFoundation\Team Explorer\Git\cmd\git.exe

C:\temp> git config --list
error: cannot spawn less: No such file or directory
core.symlinks=false
core.autocrlf=true
color.diff=auto
color.status=auto
color.branch=auto
color.interactive=true
pack.packsizelimit=2g
help.format=html
http.sslcainfo=/ssl/certs/ca-bundle.crt
diff.astextplain.textconv=astextplain
rebase.autosquash=true
filter.lfs.clean=git-lfs clean -- %f
filter.lfs.smudge=git-lfs smudge -- %f
filter.lfs.process=git-lfs filter-process
filter.lfs.required=true
include.path=C:/Program Files (x86)/Git/etc/gitconfig
include.path=C:/Program Files/Git/etc/gitconfig
core.repositoryformatversion=0
core.filemode=false
core.bare=false
core.logallrefupdates=true
core.symlinks=false
core.ignorecase=true
remote.origin.url=ssh://git@git.testgit.com:5000/testusr/test-prj.git
remote.origin.fetch=+refs/heads/*:refs/remotes/origin/*
branch.main.remote=origin
branch.main.merge=refs/heads/main

따라서 이런 경우에는 명시적으로 false로 설정해야 합니다.

C:\temp> git config --global core.autocrlf false

이렇게 환경 설정이 되면, 윈도우에서 get fetch해도 해당 .sh 파일을 wsl(또는 리눅스) 환경에서 정상적으로 실행할 수 있습니다.

하지만 개인적으로 저 설정을 다른 개발자 PC에서 일일이 하는 것은 다소 불편한 해결책으로 보입니다. 그보다는, sh 파일에 대한 개행을 특별히 LF만으로 처리할 것이기 때문에 ".gitattributes" 파일을 통해 개행 자동 관리에서 제외하는 설정이 더 나은 듯합니다.

# .gitattributes

*.sh     -text

(참고로 이렇게 설정 후, 타 PC의 'git fetch'에 반영하려면 대상 파일이 당연히 변경되어야 합니다. 그렇지 않으면 "git fetch"로는 변경 사항이 없으므로 다시 내려받지 않아 기존 파일이 유지될 뿐입니다.)




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 12/13/2021]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 46  47  48  49  50  [51]  52  53  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12664정성태6/9/202115305오류 유형: 723. COM+ PIA 참조 시 "This operation failed because the QueryInterface call on the COM component" 오류
12663정성태6/9/202117692.NET Framework: 1065. Windows Forms - 속성 창의 디자인 설정 지원: 문자열 목록 내에서 항목을 선택하는 TypeConverter 제작파일 다운로드1
12662정성태6/8/202115415.NET Framework: 1064. C# COM 개체를 PIA(Primary Interop Assembly)로써 "Embed Interop Types" 참조하는 방법파일 다운로드1
12661정성태6/4/202127421.NET Framework: 1063. C# - MQTT를 이용한 클라이언트/서버(Broker) 통신 예제 [4]파일 다운로드1
12660정성태6/3/202118143.NET Framework: 1062. Windows Forms - 폼 내에서 발생하는 마우스 이벤트를 자식 컨트롤 영역에 상관없이 수신하는 방법 [1]파일 다운로드1
12659정성태6/2/202119035Linux: 40. 우분투 설치 후 MBR 디스크 드라이브 여유 공간이 인식되지 않은 경우 - Logical Volume Management
12658정성태6/2/202116820Windows: 194. Microsoft Store에 있는 구글의 공식 Youtube App
12657정성태6/2/202117626Windows: 193. 윈도우 패키지 관리자 - winget 설치
12656정성태6/1/202115943.NET Framework: 1061. 서버 유형의 COM+에 적용할 수 없는 Server GC
12655정성태6/1/202114594오류 유형: 722. windbg/sos - savemodule - Fail to read memory
12654정성태5/31/202115260오류 유형: 721. Hyper-V - Saved 상태의 VM을 시작 시 오류 발생
12653정성태5/31/202118507.NET Framework: 1060. 닷넷 GC에 새롭게 구현되는 DPAD(Dynamic Promotion And Demotion for GC)
12652정성태5/31/202115973VS.NET IDE: 164. Visual Studio - Web Deploy로 Publish 시 암호창이 매번 뜨는 문제
12651정성태5/31/202116181오류 유형: 720. PostgreSQL - ERROR: 22P02: malformed array literal: "..."
12650정성태5/17/202115509기타: 82. OpenTabletDriver의 버튼에 더블 클릭을 매핑 및 게임에서의 지원 방법
12649정성태5/16/202117702.NET Framework: 1059. 세대 별 GC(Garbage Collection) 방식에서 Card table의 사용 의미 [1]
12648정성태5/16/202116468사물인터넷: 66. PC -> FTDI -> NodeMCU v1 ESP8266 기기를 UART 핀을 연결해 직렬 통신하는 방법파일 다운로드1
12647정성태5/15/202116723.NET Framework: 1058. C# - C++과의 연동을 위한 구조체의 fixed 배열 필드 사용파일 다운로드1
12646정성태5/15/202115529사물인터넷: 65. C# - Arduino IDE의 Serial Monitor 기능 구현파일 다운로드1
12645정성태5/14/202115581사물인터넷: 64. NodeMCU v1 ESP8266 - LittleFS를 이용한 와이파이 접속 정보 업데이트파일 다운로드1
12644정성태5/14/202116852오류 유형: 719. 윈도우 - 제어판의 "프로그램 및 기능" / "Windows 기능 켜기/끄기" 오류 0x800736B3
12643정성태5/14/202116817오류 유형: 718. 서버 유형의 COM+ 사용 시 0x80080005(Server execution failed) 오류 발생
12642정성태5/14/202118513오류 유형: 717. The 'Microsoft.ACE.OLEDB.12.0' provider is not registered on the local machine.
12641정성태5/13/202117317디버깅 기술: 179. 윈도우용 .NET Core 3 이상에서 Windbg의 sos 사용법
12640정성태5/13/202120913오류 유형: 716. RDP 연결 - Because of a protocol error (code: 0x112f), the remote session will be disconnected. [1]
12639정성태5/12/202117321오류 유형: 715. Arduino: Open Serial Monitor - The module '...\detection.node' was compiled against a different Node.js version using NODE_MODULE_VERSION
... 46  47  48  49  50  [51]  52  53  54  55  56  57  58  59  60  ...