Microsoft MVP성태의 닷넷 이야기
.NET Framework: 2070. .NET 7 - Console.ReadKey와 리눅스의 터미널 타입 [링크 복사], [링크+제목 복사],
조회: 13003
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

.NET 7 - Console.ReadKey와 리눅스의 터미널 타입

다음의 글이 있군요. ^^

Console.ReadKey improvements in .NET 7
; https://devblogs.microsoft.com/dotnet/console-readkey-improvements-in-net-7/

Console.ReadKey 구현 코드를 리눅스를 제대로 지원하기 위해 완전히 갈아엎었다고 합니다. 위의 글에 보면, 리눅스의 터미널 타입이 매우 다양(xterm, xterm-256color, linux,...)하게 있는 것을 볼 수 있는데요, 참고로 WSL 인스턴스에 대한 터미널 타입은,

$ echo $TERM
xterm-256color

이렇게 볼 수 있습니다. 아울러 이에 대한 차이점을 위의 글에서 showkey 프로그램을 이용해 설명하고 있습니다. 예를 들어, SSH 접속 클라이언트가 (WSL Shell과 동일한 타입인) Visual Studio Code인 경우,

// Visual Studio Code 또는 WSL 환경인 경우

$ echo $TERM
xterm-256color

$ showkey -a
^[OQ     27 0033 0x1b  <==== F2 키를 누른 경우
         79 0117 0x4f
         81 0121 0x51
^[[H     27 0033 0x1b  <=== Home 키를 누른 경우
         91 0133 0x5b
         72 0110 0x48
^[[F     27 0033 0x1b  <=== End 키를 누른 경우
         91 0133 0x5b
         70 0106 0x46

// F2 값에 대한 바이트 해석
^[ == ESC == 0x1b
O == 0x4f
Q == 0x51

위와 같은 바이트 배열로 구성되는 반면 (기본 모드가 xterm, ESC[n~]으로 설정된) putty를 사용하면,

// PuTTY로 접속한 경우

$ echo $TERM
xterm

$ showkey -a
^[[12~   27 0033 0x1b  <==== F2 키를 누른 경우
         91 0133 0x5b
         49 0061 0x31
         50 0062 0x32
        126 0176 0x7e
^[[1~    27 0033 0x1b  <=== Home 키를 누른 경우
         91 0133 0x5b
         49 0061 0x31
        126 0176 0x7e
^[[4~    27 0033 0x1b  <=== End 키를 누른 경우
         91 0133 0x5b
         52 0064 0x34
        126 0176 0x7e

// F2 값에 대한 바이트 해석
^[ == ESC == 0x1b
[ == 0x5b
1 == 0x31
2 == 0x32
~ == 0x7e

위와 같이 나옵니다. 실제로 해당 클라이언트 프로그램이 처리하는 키 입력이 showkey에서 보여주는 모드와 정확하게 일치해서 처리가 됩니다. 이에 대해 간단하게 테스트를 해볼까요? ^^

cat 명령어를 이용해 직접 키보드로부터 입력받은 데이터를 파일로 쓰는 명령어를 실행하면,

// cat 명령 후,
// Home, Enter, Ctrl+D 키를 차례대로 입력

$ cat > test.txt
^[[1~
$ 

위의 결과가 터미널 타입에 따라 달라집니다. putty 콘솔에서 위의 명령어를 실행하면, test.txt의 파일은 5바이트가 되고, Visual Studio Code 또는 WSL 환경에서 실행해 보면 4바이트가 나옵니다.

왜냐하면, 위에서의 showkey 결과를 반영하기 때문입니다.

// putty의 경우, 저장된 test.txt의 hexa 값을 보면,
HOME 키에 해당하는 0x1b, 0x5b, 0x31, 0x73과 Enter 키의 개행 값인 0x0a를 포함해 5바이트

// Visual Studio Code 또는 WSL 환경에서 저장된 test.txt의 hexa 값을 보면,
HOME 키에 해당하는 0x1b, 0x5b, 0x48과 Enter 키의 개행 값인 0x0a를 포함해 4바이트

저런 식으로 터미널 유형에 따른 차이가 발생하는 모든 경우의 수를 기존의 윈도우에 맞춰진 Console.ReadKey 코드로 구현하기 버거웠던 것이고, .NET 7에 포함된 System.Console 모듈에 포함된 Console.ReadKey부터는 리눅스의 tcgetattr, tcsetattr, read의 sys-call을 활용함으로써 개선을 한 것입니다.

이와 관련한 실제 사례가 이슈로 있는데요,

Console.ReadKey and pressing SHIFT+END returns invalid escape sequence on WSL/Ubuntu 
; https://github.com/dotnet/runtime/issues/45597

using System;

class Program
{
    static void Main(string[] args)
    {
        while (true)
        {
            var key = Console.ReadKey(true);
            Console.WriteLine($"Key: {key.Key} Modifiers: {key.Modifiers} Char: {(key.KeyChar < ' ' || (int)key.KeyChar >= 126 ? "0x" + ((int)key.KeyChar).ToString("x2") : key.KeyChar.ToString())}");
        }
    }
}

위의 프로그램을 .NET 6 환경에서 실행하면 Shift + END 키의 경우 다음과 같이 이상한 출력 결과를 얻게 되지만,

$ dotnet ./ConsoleApp2.dll
Key: Escape Modifiers: 0 Char: 0x1b
Key: D1 Modifiers: 0 Char: 1
Key: 0 Modifiers: 0 Char: ;
Key: D2 Modifiers: 0 Char: 2
Key: F Modifiers: Shift Char: F

$ showkey -a
^[[1;2F  27 0033 0x1b   <== Shift + End 키를 누른 경우
         91 0133 0x5b
         49 0061 0x31
         59 0073 0x3b
         50 0062 0x32
         70 0106 0x46

동일한 프로그램을 .NET 7 환경에서 실행해 보면,

$ ~/mydot/dotnet ./ConsoleApp2.dll
Key: End Modifiers: Shift Char: 0x00

정확하게 해석해냅니다. ^^




그나저나, 개인적으로 리알못이라 이해가 잘 안 되는 부분이 있는데요. ^^ 아래와 같은 글을 보면,

Set a terminal type or terminal emulation
; https://kb.iu.edu/d/acpy

터미널 타입을 단순히 TERM 환경 변수에 값을 주는 것만으로 변경할 수 있는 것 같은데요, 예를 들어 WSL에서 TERM을 vt100으로 바꾸면,

$ export TERM=vt100

showkey의 결과가 달라져야 할 것 같은데, 그대로입니다.

$ showkey -a

Press any keys - Ctrl-D will terminate this program

^[OQ     27 0033 0x1b  <==== F2 키를 누른 경우
         79 0117 0x4f
         81 0121 0x51
^[[H     27 0033 0x1b  <=== Home 키를 누른 경우
         91 0133 0x5b
         72 0110 0x48
^[[F     27 0033 0x1b  <=== End 키를 누른 경우
         91 0133 0x5b
         70 0106 0x46

바꿀 수 없는 건가요? 아니면 바꾸는 또 다른 방법이 있는 걸까요? ^^ (아시는 분은 덧글 부탁드립니다.)

참고로, PuTTY의 경우 "Connection" / "Data"의 "Terminal-type string"과 "Terminal" / "Keyboard"의 값을 "ESC[n~", "Linux", "Xterm R6", "VT400", "VT100+", "SCO", "Xterm216+" 설정에서 바꿔서 접속하면 showkey의 결과가 달라지는 것은 확인할 수 있습니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 11/21/2022]

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

비밀번호

댓글 작성자
 




1  2  3  4  5  6  7  8  9  10  11  [12]  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13641정성태6/11/20248659Linux: 71. Ubuntu 20.04를 22.04로 업데이트
13640정성태6/10/20248830Phone: 21. C# MAUI - Android 환경에서의 파일 다운로드(DownloadManager)
13639정성태6/8/20248436오류 유형: 906. C# MAUI - Android Emulator에서 "Waiting For Debugger"로 무한 대기
13638정성태6/8/20248520오류 유형: 905. C# MAUI - 추가한 layout XML 파일이 Resource.Layout 멤버로 나오지 않는 문제
13637정성태6/6/20248446Phone: 20. C# MAUI - 유튜브 동영상을 MediaElement로 재생하는 방법
13636정성태5/30/20248087닷넷: 2264. C# - 형식 인자로 인터페이스를 갖는 제네릭 타입으로의 형변환파일 다운로드1
13635정성태5/29/20248936Phone: 19. C# MAUI - 안드로이드 "Share" 대상으로 등록하는 방법
13634정성태5/24/20249416Phone: 18. C# MAUI - 안드로이드 플랫폼에서의 Activity 제어 [1]
13633정성태5/22/20248943스크립트: 64. 파이썬 - ASGI를 만족하는 최소한의 구현 코드
13632정성태5/20/20248559Phone: 17. C# MAUI - Android 내에 Web 서비스 호스팅
13631정성태5/19/20249320Phone: 16. C# MAUI - /Download 등의 공용 디렉터리에 접근하는 방법 [1]
13630정성태5/19/20248863닷넷: 2263. C# - Thread가 Task보다 더 빠르다는 어떤 예제(?)
13629정성태5/18/20249159개발 환경 구성: 710. Android - adb.exe를 이용한 파일 전송
13628정성태5/17/20248540개발 환경 구성: 709. Windows - WHPX(Windows Hypervisor Platform)를 이용한 Android Emulator 가속
13627정성태5/17/20248603오류 유형: 904. 파이썬 - UnicodeEncodeError: 'ascii' codec can't encode character '...' in position ...: ordinal not in range(128)
13626정성태5/15/20248867Phone: 15. C# MAUI - MediaElement Source 경로 지정 방법파일 다운로드1
13625정성태5/14/20248923닷넷: 2262. C# - Exception Filter 조건(when)을 갖는 catch 절의 IL 구조
13624정성태5/12/20248720Phone: 14. C# - MAUI에서 MediaElement 사용파일 다운로드1
13623정성태5/11/20248418닷넷: 2261. C# - 구글 OAuth의 JWT (JSON Web Tokens) 해석파일 다운로드1
13622정성태5/10/20249205닷넷: 2260. C# - Google 로그인 연동 (ASP.NET 예제)파일 다운로드1
13621정성태5/10/20248636오류 유형: 903. IISExpress - Failed to register URL "..." for site "..." application "/". Error description: Cannot create a file when that file already exists. (0x800700b7)
13620정성태5/9/20248542VS.NET IDE: 190. Visual Studio가 node.exe를 경유해 Edge.exe를 띄우는 경우
13619정성태5/7/20248860닷넷: 2259. C# - decimal 저장소의 비트 구조파일 다운로드1
13618정성태5/6/20248653닷넷: 2258. C# - double (배정도 실수) 저장소의 비트 구조파일 다운로드1
13617정성태5/5/20249471닷넷: 2257. C# - float (단정도 실수) 저장소의 비트 구조파일 다운로드1
13616정성태5/3/20248617닷넷: 2256. ASP.NET Core 웹 사이트의 HTTP/HTTPS + Dual mode Socket (IPv4/IPv6) 지원 방법파일 다운로드1
1  2  3  4  5  6  7  8  9  10  11  [12]  13  14  15  ...