Microsoft MVP성태의 닷넷 이야기
오류 유형: 773. shell script 실행 시 "$'\r': command not found" 오류 [링크 복사], [링크+제목 복사],
조회: 13999
글쓴 사람
정성태 (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

비밀번호

댓글 작성자
 




... 151  [152]  153  154  155  156  157  158  159  160  161  162  163  164  165  ...
NoWriterDateCnt.TitleFile(s)
1252정성태3/1/201225372Windows: 55. 윈도우 8 베타 설치 과정 [1]
1251정성태2/27/201229311VC++: 60. C/C++ Native 스레드 콜 스택 덤프를 얻는 공개 라이브러리 [2]파일 다운로드1
1250정성태2/27/201231434VC++: 59. C/C++ 프로젝트 빌드 속도 개선 - UnityBuild를 아세요? [3]
1249정성태2/26/201231118.NET Framework: 311. .NET 스레드 콜 스택 덤프 (5) - ICorDebug 인터페이스 사용법 [2]파일 다운로드3
1248정성태2/25/201242625.NET Framework: 310. C#의 Shift 비트 연산 정리파일 다운로드1
1247정성태2/25/201225290.NET Framework: 309. .NET 응용 프로그램에 기본 생성되는 스레드들에 대한 탐구 [1]파일 다운로드1
1246정성태2/25/201224875개발 환경 구성: 145. 한영 변환은 되지만, 정작 한글 입력이 안되는 경우
1245정성태2/25/201235586개발 환경 구성: 144. 윈도우에서도 유닉스처럼 명령행으로 원격 접속하는 방법
1244정성태2/24/201232779.NET Framework: 308. .NET System.Threading.Thread 개체에서 Native Thread Id를 구할 수 있을까? [1]파일 다운로드1
1243정성태2/23/201232736개발 환경 구성: 143. Visual Studio 2010 - .NET Framework 소스 코드 디버깅 - 두 번째 이야기 [1]
1242정성태2/20/201239594VC++: 58. API Hooking - 64비트를 고려해야 한다면? EasyHook! [7]파일 다운로드1
1241정성태2/20/201226420.NET Framework: 307. .NET 4.0 응용 프로그램을 위한 ILMerge
1240정성태2/19/201232709디버깅 기술: 48. C/C++ JNI DLL을 Visual Studio로 디버깅하는 방법 [2]
1239정성태2/19/201224325.NET Framework: 306. 컴퓨터에 실행된 프로세스 중에 닷넷 응용 프로그램임을 알 수 있는 방법 - C# [1]파일 다운로드1
1238정성태2/19/201228186.NET Framework: 305. GetPrivateProfileSection / WritePrivateProfileSection의 C# 버전파일 다운로드1
1237정성태2/18/201232526개발 환경 구성: 142. Windows Embedded POSReady 7 설치 [1]
1236정성태2/17/201228334개발 환경 구성: 141. Windows 2008 R2 RDP 라이선스 서버 설치하는 방법
1235정성태2/16/201226781.NET Framework: 304. Hyper-V의 가상 머신을 C#으로 제어하는 방법 [1]파일 다운로드1
1234정성태2/16/201227180.NET Framework: 303. 원본 파일의 공백/라인을 유지한 체 XML 파일을 저장하는 방법 [1]파일 다운로드1
1233정성태2/16/201233305.NET Framework: 302. supportedRuntime 옵션과 System.BadImageFormatException 예외 [5]
1232정성태2/9/201229211VC++: 57. 웹 브라우저에서 Flash만 빼고 다른 ActiveX를 차단할 수 있을까? [3]파일 다운로드1
1231정성태2/8/201238731VC++: 56. Win32 API 후킹 - Trampoline API Hooking [5]파일 다운로드1
1230정성태2/6/201224070개발 환경 구성: 140. 프로젝트 생성 시부터 "Enable the Visual Studio hosting process" 옵션을 끄는 방법
1229정성태2/4/201229092.NET Framework: 301. P/Invoke의 성능을 높이기 위해 C++/CLI가 선택되려면? [5]파일 다운로드1
1228정성태2/4/201278383.NET Framework: 300. C#으로 만드는 음성인식/TTS 프로그램 [47]파일 다운로드1
1227정성태2/3/201229268.NET Framework: 299. 해당 어셈블리가 Debug 빌드인지, Release 빌드인지 알아내는 방법파일 다운로드1
... 151  [152]  153  154  155  156  157  158  159  160  161  162  163  164  165  ...