Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일

(시리즈 글이 14개 있습니다.)
.NET Framework: 292. RSACryptoServiceProvider의 공개키와 개인키 구분
; https://www.sysnet.pe.kr/2/0/1218

.NET Framework: 327. RSAParameters와 System.Numerics.BigInteger 이야기
; https://www.sysnet.pe.kr/2/0/1295

.NET Framework: 329. C# - Rabin-Miller 소수 생성방법을 이용하여 RSACryptoServiceProvider의 개인키를 직접 채워보자
; https://www.sysnet.pe.kr/2/0/1300

.NET Framework: 356. (공개키를 담은) 자바의 key 파일을 닷넷의 RSACryptoServiceProvider에서 사용하는 방법
; https://www.sysnet.pe.kr/2/0/1401

.NET Framework: 383. RSAParameters의 ToXmlString과 ExportParameters의 결과 비교
; https://www.sysnet.pe.kr/2/0/1491

.NET Framework: 565. C# - Rabin-Miller 소수 생성 방법을 이용하여 RSACryptoServiceProvider의 개인키를 직접 채워보자 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/10925

.NET Framework: 566. openssl의 PKCS#1 PEM 개인키 파일을 .NET RSACryptoServiceProvider에서 사용하는 방법
; https://www.sysnet.pe.kr/2/0/10926

.NET Framework: 638. RSAParameters와 RSA
; https://www.sysnet.pe.kr/2/0/11140

.NET Framework: 1037. openssl의 PEM 개인키 파일을 .NET RSACryptoServiceProvider에서 사용하는 방법 (2)
; https://www.sysnet.pe.kr/2/0/12598

.NET Framework: 2093. C# - PKCS#8 PEM 파일을 이용한 RSA 개인키/공개키 설정 방법
; https://www.sysnet.pe.kr/2/0/13245

닷넷: 2297. C# - ssh-keygen으로 생성한 Public Key 파일 해석과 fingerprint 값(md5, sha256) 생성
; https://www.sysnet.pe.kr/2/0/13739

닷넷: 2297. C# - ssh-keygen으로 생성한 ecdsa 유형의 Public Key 파일 해석
; https://www.sysnet.pe.kr/2/0/13742

닷넷: 2300. C# - OpenSSH의 공개키 파일에 대한 "BEGIN OPENSSH PUBLIC KEY" / "END OPENSSH PUBLIC KEY" PEM 포맷
; https://www.sysnet.pe.kr/2/0/13747

닷넷: 2302. C# - ssh-keygen으로 생성한 Private Key와 Public Key 연동
; https://www.sysnet.pe.kr/2/0/13749




C# - OpenSSH의 공개키 파일에 대한 "BEGIN OPENSSH PUBLIC KEY" / "END OPENSSH PUBLIC KEY" PEM 포맷

근래의 ssh-keygen에서는 키 파일을 생성하는 경우 개인키 파일의 포맷이 "OPENSSH PRIVATE KEY" 구분자의 PEM 파일로 나옵니다.

// 테스트 편의상 -N "" 옵션으로 passphrase를 빈 문자열로 설정하였습니다.

c:\temp> ssh-keygen -N "" -t rsa -b 4096 -f test_rsa
...[생략]...

c:\temp> type test_rsa
-----BEGIN OPENSSH PRIVATE KEY-----
b3BlbnNzaC1rZXktdjEAAAAABG5vbmUAAAAEbm9uZQAAAAAAAAABAAACFwAAAAdzc2gtcn
...[생략]...
KN3bndgw7MlZwRCUu3PBopxuL19RWbJVMtcMBLwRHfSDHHFvtpzLPYdEeHItOK+HCv9/bD
JDyAnpeeQoQd2QAAAAtrZXZpbkBUSEUxMgECAwQFBg==
-----END OPENSSH PRIVATE KEY-----

반면, 공개키 파일은 PEM 형식으로 나오지 않는데,

c:\temp> type test_rsa.pub
ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAACAQCvnNE7kvQyCLJIi3i1hZwVmLNheUb8E1oc3R94YLcnOKlisbuXx3hiSGjOPOx9Uedf/Pp73bx8Otu/9VudriZ910cPTzDDR6zJPfUYHyDiltwJ3zcKpkoG6z6ilIJVuX1Cm8S9q+pwkVOm7ij+FSiF8R/WxlblPMZzdTRyCYMCiOJ0KjZXU8as3iGadnQMZD+WUn6t/gPUKfIvCw1uAIPY5KDqp8RJeeMeJJw55RCkkjHzv1ghYvQDMCqpWXQ+OT5n1DQEpbH1994UYBJhpJC/W95+Tn2pqTPmcdobP7+nl3COVwxB77uk9Hxkfr9ldBxoZiEHchhm1qDUCp82QGFbImufQ3A2wYdQIyRTHD11gro17jj+5U7ae5szeRQzlh6RWaxUvPefFvs6nKhCXOE5yT5Ss5QoMBjivyUWdcMPW2X15dB0RAR03HA1GmyTnMj2zVJh/7p0hJR11U5EcOVQ969RfyPf7GvZZlhMSaYNdLiwg4O1f8FjS5wk+O7srimgnSe+K9Mz2qorCosYp4nZq/dduuy56UK2j609JbXXtfRRN0k16/JeDQvP7S1Pv3zQtwDvyBxGRlZ7UKhkAC1R0HRGjqWwqtVOTLtpKND64ENGXSYSddesF9SCBK2dwRjyhSLuv74dkpHzGbNAk0gTW1mnChJN6rHIrBtCA4rFRQ== testusr@TESTPC

이전 포맷(PKCS#1, PKCS#8)에서와는 다르게 위의 "BEGIN/END OPENSSH PRIVATE KEY" 형식을 PEM 형식으로 변환하는 방법을 ssh-keygen 및 openssl은 제공하지 않습니다. (제가 방법을 모르는 것일 수도 있습니다. ^^;)




재미있는 건, "BEGIN OPENSSH PUBLIC KEY" / "END OPENSSH PUBLIC KEY" 구분자를 가진 공개키 파일에 대한 검색도 잘 안됩니다. 한 가지 찾은 것이 다음의 글인데요,

Justin Freid’s PGP & SSH Public Keys
; https://justinfreid.com/justin-freids-pgp-ssh-public-keys/

위에서는 단지 "BEGIN OPENSSH PUBLIC KEY" / "END OPENSSH PUBLIC KEY" 구분자 사이에 OpenSSH의 (rfc4253 포맷으로 출력된) 공개키를 그대로 복사해 넣은 유형으로 보여줍니다.

-----BEGIN OPENSSH PUBLIC KEY-----
ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAACAQCvnNE7kvQyCLJIi3i1hZwVmLNheUb8E1
oc3R94YLcnOKlisbuXx3hiSGjOPOx9Uedf/Pp73bx8Otu/9VudriZ910cPTzDDR6zJPfUY
HyDiltwJ3zcKpkoG6z6ilIJVuX1Cm8S9q+pwkVOm7ij+FSiF8R/WxlblPMZzdTRyCYMCiO
J0KjZXU8as3iGadnQMZD+WUn6t/gPUKfIvCw1uAIPY5KDqp8RJeeMeJJw55RCkkjHzv1gh
YvQDMCqpWXQ+OT5n1DQEpbH1994UYBJhpJC/W95+Tn2pqTPmcdobP7+nl3COVwxB77uk9H
xkfr9ldBxoZiEHchhm1qDUCp82QGFbImufQ3A2wYdQIyRTHD11gro17jj+5U7ae5szeRQz
lh6RWaxUvPefFvs6nKhCXOE5yT5Ss5QoMBjivyUWdcMPW2X15dB0RAR03HA1GmyTnMj2zV
Jh/7p0hJR11U5EcOVQ969RfyPf7GvZZlhMSaYNdLiwg4O1f8FjS5wk+O7srimgnSe+K9Mz
2qorCosYp4nZq/dduuy56UK2j609JbXXtfRRN0k16/JeDQvP7S1Pv3zQtwDvyBxGRlZ7UK
hkAC1R0HRGjqWwqtVOTLtpKND64ENGXSYSddesF9SCBK2dwRjyhSLuv74dkpHzGbNAk0gT
W1mnChJN6rHIrBtCA4rFRQ== testusr@TESTPC
-----END OPENSSH PUBLIC KEY-----

그다음, python으로 된 소스코드를 하나 찾은 것에도,

Generate ed25515 OpenSSH keys with Python3 cryptography library
; https://gist.github.com/thebluesnevrdie/f1522bf5305d561f42dac25ff398f2e9

#### pip install cryptography
from cryptography.hazmat.primitives.asymmetric import rsa, ed25519
from cryptography.hazmat.primitives import serialization, hashes

private_key = ed25519.Ed25519PrivateKey.generate()
public_key = private_key.public_key()

our_private = private_key.private_bytes(
    encoding=serialization.Encoding.PEM,
    format=serialization.PrivateFormat.OpenSSH,
    encryption_algorithm=serialization.NoEncryption(),
)

our_public = public_key.public_bytes(
    encoding=serialization.Encoding.OpenSSH, format=serialization.PublicFormat.OpenSSH
)

print(our_private.decode("utf-8"))
print("-----BEGIN OPENSSH PUBLIC KEY-----")
print(our_public.decode("utf-8"))
print("-----END OPENSSH PUBLIC KEY-----")

실제로 실행해 보면 public key에 대해 이런 식으로 나옵니다.

-----BEGIN OPENSSH PRIVATE KEY-----
b3BlbnNzaC1rZXktdjEAAAAABG5vbmUAAAAEbm9uZQAAAAAAAAABAAAAMwAAAAtzc2gtZWQyNTUx
OQAAACDQ3a1hJeJfZzky2WZPt4FbX7poA8TV/E9FxF+WDpBPQwAAAIj6Yds3+mHbNwAAAAtzc2gt
ZWQyNTUxOQAAACDQ3a1hJeJfZzky2WZPt4FbX7poA8TV/E9FxF+WDpBPQwAAAEB/KFTh1ivE0ozW
JF4qq7nNWeU/U+UJIbo8cWLJqnsXZtDdrWEl4l9nOTLZZk+3gVtfumgDxNX8T0XEX5YOkE9DAAAA
AAECAwQF
-----END OPENSSH PRIVATE KEY-----

-----BEGIN OPENSSH PUBLIC KEY-----
ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAINDdrWEl4l9nOTLZZk+3gVtfumgDxNX8T0XEX5YOkE9D
-----END OPENSSH PUBLIC KEY-----

이 결과는 "Justin Freid’s PGP & SSH Public Keys" 글에서의 표현과 같습니다. (단지, 마지막에 comment만 없는 것이 다릅니다.)




사실 PEM(Privacy Enhanced Mail) 형식이라는 것은 일종의 컨테이너에 불과합니다.

PKCS#1 and PKCS#8 format for RSA private key [closed]
; https://stackoverflow.com/questions/48958304/pkcs1-and-pkcs8-format-for-rsa-private-key

How to get .pem file from .key and .crt files?
; https://stackoverflow.com/questions/991758/how-to-get-pem-file-from-key-and-crt-files

RFC 1422 문서로 복잡하게 명시하고 있지만, 간략하게는 다음과 같은 정도의 형식만 가지면 PEM 형식이라고 할 수 있습니다.

  • a line consisting of 5 hyphens, the word BEGIN, one or a few (space-separated) words defining the type of data, and 5 hyphens
  • an optional (and rare) rfc822-style header, terminated by an empty line
  • base64 of the data, broken into lines of 64 characters (except the last); some programs instead use the (slightly newer) MIME limit of 76 characters
  • a line like the BEGIN line but with END instead

따라서, "BEGIN OPENSSH PUBLIC KEY" / "END OPENSSH PUBLIC KEY" 구분자를 가진 공개키 파일의 PEM 형식은 OpenSSL 측에서 정의해 그렇게 사용하면 그만입니다.

단지, 중요한 것은 그 내부의 content 포맷까지 합의가 되었는지에 대해서는 다른 문제입니다. 가령, 구분자 문자열이 "BEGIN/END RSA PRIVATE KEY"인 경우에는 내부 content가 PKCS#1 형식이어야 하고, "BEGIN/END PRIVATE KEY"인 경우에는 PKCS#8 형식이어야 합니다.

반면, OpenSSL 측이 "BEGIN/END OPENSSH PUBLIC KEY"에 대해 내부 포맷을 rfc4253 방식으로 정의했다고 가정하면 이전에 설명한 방법으로 해석하면 됩니다. 예를 들어 볼까요?

-----BEGIN OPENSSH PUBLIC KEY-----
ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAID7ttyIQ8COKmzPeAA5YhMu1IqRvOME3kgAlybbTBB9q
-----END OPENSSH PUBLIC KEY-----

우선 ed25519는 EdDSA 서명 알고리즘의 변형이라고 하는데요, "작은 공개키(32 또는 57 바이트)"라는 표현에 따라 단일 값으로 보입니다. 따라서 포맷 자체는 이렇게 간단할 듯한데요,

string  "algorithm_name"
int     P

실제로 "AAAAC3NzaC1lZDI1NTE5AAAAID7ttyIQ8COKmzPeAA5YhMu1IqRvOME3kgAlybbTBB9q" 값을 분해해 보면 다음의 값이 나옵니다.

"ssh-ed25519"
byte [32]

자, 그럼 이것을 사용해 봐야 할 텐데, 아쉽게도 .NET Cryptography 어셈블리에는 EdDSA 알고리즘을 지원하는 타입이 없습니다. 검색해 보면,

EdDSA for JWT Signing in .NET Core
; https://www.scottbrady91.com/c-sharp/eddsa-for-jwt-signing-in-dotnet-core

어쩔 수 없이 다시 Bouncy Castle을 사용해 대충 이렇게 예제 코드를 만들 수 있습니다. ^^

using Org.BouncyCastle.Crypto.Parameters;
using Org.BouncyCastle.Crypto.Signers;
using System.Text;

namespace ConsoleApp1;

// Install-Package BouncyCastle.Cryptography
internal class Program
{
    static void Main(string[] args)
    {
        string text = File.ReadAllText("test_rsa.pub.pem");

        {
            text = text.Replace("-----BEGIN OPENSSH PUBLIC KEY-----", "")
                .Replace("-----END OPENSSH PUBLIC KEY-----", "").Trim();

            string[] sshKeys = text.Split(' ');
            if (sshKeys.Length < 2)
            {
                throw new Exception("Invalid SSH Key");
            }

            string keyType = sshKeys[0];
            string comment = (sshKeys.Length >= 3) ? sshKeys[2] : "";
            byte[] bytesEncoded = Convert.FromBase64String(sshKeys[1]);

            if (keyType != "ssh-ed25519")
            {
                throw new NotSupportedException($"Unsupported key type: {keyType}");
            }

            (string algorithmName, byte[] publicKey) = DecodeSSHPublicKey(bytesEncoded);

            Ed25519PublicKeyParameters publicKeyParam = new Ed25519PublicKeyParameters(publicKey);

            Ed25519Signer signer = new Ed25519Signer();
            signer.Init(false, publicKeyParam);

            // verify signature...
        }
    }

    private static (string algorithmName, byte[] publicKey) DecodeSSHPublicKey(byte[] bytesEncoded)
    {
        string algorithmName;
        byte[] publicKey;

        using (var stream = new MemoryStream(bytesEncoded))
        using (var reader = new BinaryReader(stream))
        {
            int algorithmLength = reader.ReadInt32BE();
            algorithmName = Encoding.ASCII.GetString(reader.ReadBytes(algorithmLength));

            int length = reader.ReadInt32BE();
            publicKey = reader.ReadBytes(length);
        }

        return (algorithmName, publicKey);
    }
}

public static class BinaryReaderExtension
{
    public static Int32 ReadInt32BE(this BinaryReader reader)
    {
        byte[] bytes = new byte[4];
        bytes[3] = (byte)reader.ReadByte();
        bytes[2] = (byte)reader.ReadByte();
        bytes[1] = (byte)reader.ReadByte();
        bytes[0] = (byte)reader.ReadByte();
        return BitConverter.ToInt32(bytes);
    }
}

혹시, OpenSSH/OpenSSL 관련 코드를 잘 아시는 분이 있다면... ^^ "BEGIN OPENSSH PUBLIC KEY" / "END OPENSSH PUBLIC KEY" 파일이 정말 정의된 것인지, 그리고 그것의 내부 포맷이 이 글에서 설명한 것과 같은 것인지 덧글 부탁드립니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




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







[최초 등록일: ]
[최종 수정일: 9/28/2024]

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)
13262정성태2/15/202313506디버깅 기술: 191. dnSpy를 이용한 (소스 코드가 없는) 닷넷 응용 프로그램 디버깅 방법 [1]
13261정성태2/15/202312982Windows: 224. Visual Studio - 영문 폰트가 Fullwidth Latin Character로 바뀌는 문제
13260정성태2/14/202312222오류 유형: 847. ilasm.exe 컴파일 오류 - error : syntax error at token '-' in ... -inf
13259정성태2/14/202312172.NET Framework: 2095. C# - .NET5부터 도입된 CollectionsMarshal
13258정성태2/13/202312949오류 유형: 846. .NET Framework 4.8 Developer Pack 설치 실패 - 0x81f40001
13257정성태2/13/202312356.NET Framework: 2094. C# - Job에 Process 포함하는 방법 [1]파일 다운로드1
13256정성태2/10/202312484개발 환경 구성: 665. WSL 2의 네트워크 통신 방법 - 두 번째 이야기
13255정성태2/10/202312701오류 유형: 845. gihub - windows2022 이미지에서 .NET Framework 4.5.2 미만의 프로젝트에 대한 빌드 오류
13254정성태2/10/202312587Windows: 223. (WMI 쿼리를 위한) PowerShell 문자열 escape 처리
13253정성태2/9/202313719Windows: 222. C# - 다른 윈도우 프로그램이 실행되었음을 인식하는 방법파일 다운로드1
13252정성태2/9/202311610오류 유형: 844. ssh로 명령어 수행 시 멈춤 현상
13251정성태2/8/202312198스크립트: 44. 파이썬의 3가지 스레드 ID
13250정성태2/8/202313797오류 유형: 843. System.InvalidOperationException - Unable to configure HTTPS endpoint
13249정성태2/7/202314414오류 유형: 842. 리눅스 - You must wait longer to change your password
13248정성태2/7/202311068오류 유형: 841. 리눅스 - [사용자 계정] is not in the sudoers file. This incident will be reported.
13247정성태2/7/202312624VS.NET IDE: 180. Visual Studio - 닷넷 소스 코드 디버깅 중 "Decompile source code"가 동작하는 않는 문제
13246정성태2/6/202312448개발 환경 구성: 664. Hyper-V에 설치한 리눅스 VM의 VHD 크기 늘리는 방법 - 두 번째 이야기
13245정성태2/6/202312989.NET Framework: 2093. C# - PKCS#8 PEM 파일을 이용한 RSA 개인키/공개키 설정 방법파일 다운로드1
13244정성태2/5/202312098VS.NET IDE: 179. Visual Studio - External Tools에 Shell 내장 명령어 등록
13243정성태2/5/202313238디버깅 기술: 190. windbg - Win32 API 호출 시점에 BP 거는 방법 [1]
13242정성태2/4/202312079디버깅 기술: 189. ASP.NET Web Application (.NET Framework) 프로젝트의 숨겨진 예외 - System.UnauthorizedAccessException
13241정성태2/3/202310676디버깅 기술: 188. ASP.NET Web Application (.NET Framework) 프로젝트의 숨겨진 예외 - System.IO.FileNotFoundException
13240정성태2/1/202311354디버깅 기술: 187. ASP.NET Web Application (.NET Framework) 프로젝트의 숨겨진 예외 - System.Web.HttpException
13239정성태2/1/202310718디버깅 기술: 186. C# - CacheDependency의 숨겨진 예외 - System.Web.HttpException
13238정성태1/31/202315020.NET Framework: 2092. IIS 웹 사이트를 TLS 1.2 또는 TLS 1.3 프로토콜로만 운영하는 방법
13237정성태1/30/202314314.NET Framework: 2091. C# - 웹 사이트가 어떤 버전의 TLS/SSL을 지원하는지 확인하는 방법
... 16  17  18  19  20  21  22  23  24  25  26  [27]  28  29  30  ...