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

.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

비밀번호

댓글 작성자
 




... 16  17  18  19  20  21  22  23  24  25  26  27  [28]  29  30  ...
NoWriterDateCnt.TitleFile(s)
12933정성태1/21/20227904.NET Framework: 1136. C# - ffmpeg(FFmpeg.AutoGen)를 이용해 MP2 오디오 파일 디코딩 예제(decode_audio.c)파일 다운로드1
12932정성태1/20/20228353.NET Framework: 1135. C# - ffmpeg(FFmpeg.AutoGen)로 하드웨어 가속기를 이용한 비디오 디코딩 예제(hw_decode.c) [2]파일 다운로드1
12931정성태1/20/20226465개발 환경 구성: 632. ASP.NET Core 프로젝트를 AKS/k8s에 올리는 과정
12930정성태1/19/20227096개발 환경 구성: 631. AKS/k8s의 Volume에 파일 복사하는 방법
12929정성태1/19/20226886개발 환경 구성: 630. AKS/k8s의 Pod에 Volume 연결하는 방법
12928정성태1/18/20227017개발 환경 구성: 629. AKS/Kubernetes에서 호스팅 중인 pod에 shell(/bin/bash)로 진입하는 방법
12927정성태1/18/20226780개발 환경 구성: 628. AKS 환경에 응용 프로그램 배포 방법
12926정성태1/17/20227275오류 유형: 787. AKS - pod 배포 시 ErrImagePull/ImagePullBackOff 오류
12925정성태1/17/20227404개발 환경 구성: 627. AKS의 준비 단계 - ACR(Azure Container Registry)에 docker 이미지 배포
12924정성태1/15/20228895.NET Framework: 1134. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 비디오 디코딩 예제(decode_video.c) [2]파일 다운로드1
12923정성태1/15/20227772개발 환경 구성: 626. ffmpeg.exe를 사용해 비디오 파일을 MPEG1 포맷으로 변경하는 방법
12922정성태1/14/20226835개발 환경 구성: 625. AKS - Azure Kubernetes Service 생성 및 SLO/SLA 변경 방법
12921정성태1/14/20225793개발 환경 구성: 624. Docker Desktop에서 별도 서버에 설치한 docker registry에 이미지 올리는 방법
12920정성태1/14/20226547오류 유형: 786. Camtasia - An error occurred with the camera: Failed to Add Video Sampler.
12919정성태1/13/20226375Windows: 199. Host Network Service (HNS)에 의해서 점유되는 포트
12918정성태1/13/20226604Linux: 47. WSL - shell script에서 설정한 환경 변수가 스크립트 실행 후 반영되지 않는 문제
12917정성태1/12/20225818오류 유형: 785. C# - The type or namespace name '...' could not be found (are you missing a using directive or an assembly reference?)
12916정성태1/12/20225580오류 유형: 784. TFS - One or more source control bindings for this solution are not valid and are listed below.
12915정성태1/11/20225861오류 유형: 783. Visual Studio - We didn't find any interpreters
12914정성태1/11/20227846VS.NET IDE: 172. 비주얼 스튜디오 2022의 파이선 개발 환경 지원
12913정성태1/11/20228343.NET Framework: 1133. C# - byte * (바이트 포인터)를 FileStream으로 쓰는 방법 [1]
12912정성태1/11/20228976개발 환경 구성: 623. ffmpeg.exe를 사용해 비디오 파일의 이미지를 PGM(Portable Gray Map) 파일 포맷으로 출력하는 방법 [1]
12911정성태1/11/20226277VS.NET IDE: 171. 비주얼 스튜디오 - 더 이상 만들 수 없는 "ASP.NET Core 3.1 Web Application (.NET Framework)" 프로젝트
12910정성태1/10/20226749제니퍼 .NET: 30. 제니퍼 닷넷 적용 사례 (8) - CPU high와 DB 쿼리 성능에 문제가 함께 있는 사이트
12909정성태1/10/20228149오류 유형: 782. Visual Studio 2022 설치 시 "Couldn't install Microsoft.VisualCpp.Redist.14.Latest"
12908정성태1/10/20225986.NET Framework: 1132. C# - ref/out 매개변수의 IL 코드 처리
... 16  17  18  19  20  21  22  23  24  25  26  27  [28]  29  30  ...