Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 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# - ssh-keygen으로 생성한 Public Key 파일 해석과 fingerprint 값(md5, sha256) 생성

ssh-keygen 등으로 생성한 SSH public key의 경우, 다음과 같은 식의 포맷을 가지는데요,

// 여기서는 테스트를 쉽게 하기 위해 암호를 생략하도록 (-N 옵션의 빈 문자열) 지정했습니다. (현업에서는 암호 사용을 권장합니다.)

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

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

이 포맷을 다음의 명령어를 이용해,

// 또는, openssl을 이용.
c:\temp> ssh-keygen -f test_rsa.pub -e -m pem > test_rsa.pub.pkcs1

c:\temp> type test_rsa.pub.pkcs1
-----BEGIN RSA PUBLIC KEY-----
MIICCgKCAgEAr5zRO5L0MgiySIt4tYWcFZizYXlG/BNaHN0feGC3JzipYrG7l8d4
YkhozjzsfVHnX/z6e928fDrbv/Vbna4mfddHD08ww0esyT31GB8g4pbcCd83CqZK
Bus+opSCVbl9QpvEvavqcJFTpu4o/hUohfEf1sZW5TzGc3U0cgmDAojidCo2V1PG
rN4hmnZ0DGQ/llJ+rf4D1CnyLwsNbgCD2OSg6qfESXnjHiScOeUQpJIx879YIWL0
AzAqqVl0Pjk+Z9Q0BKWx9ffeFGASYaSQv1vefk59qakz5nHaGz+/p5dwjlcMQe+7
pPR8ZH6/ZXQcaGYhB3IYZtag1AqfNkBhWyJrn0NwNsGHUCMkUxw9dYK6Ne44/uVO
2nubM3kUM5YekVmsVLz3nxb7OpyoQlzhOck+UrOUKDAY4r8lFnXDD1tl9eXQdEQE
dNxwNRpsk5zI9s1SYf+6dISUddVORHDlUPevUX8j3+xr2WZYTEmmDXS4sIODtX/B
Y0ucJPju7K4poJ0nvivTM9qqKwqLGKeJ2av3XbrsuelCto+tPSW117X0UTdJNevy
Xg0Lz+0tT7980LcA78gcRkZWe1CoZAAtUdB0Ro6lsKrVTky7aSjQ+uBDRl0mEnXX
rBfUggStncEY8oUi7r++HZKR8xmzQJNIE1tZpwoSTeqxyKwbQgOKxUUCAwEAAQ==
-----END RSA PUBLIC KEY-----

PKCS#1 PEM 포맷으로 변환할 수 있습니다. 이런 경우라면 닷넷 5부터 추가된 ImportFromPem / ImportFromEncryptedPem 메서드로,

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

공개키를 로드하는 것이 가능합니다.

string pemText = File.ReadAllText("test_rsa.pub.pkcs1");
using (RSA rsa = RSA.Create())
{
    rsa.ImportFromPem(pemText.ToCharArray()); // 현재 PUBLIC KEY, PRIVATE KEY, RSA PRIVATE KEY, RSA PUBLIC KEY 포맷을 지원합니다.
    Console.WriteLine(rsa.ToXmlString(false));
}

혹은, test_rsa.pub.pkcs1 파일 내부의 구분자를 제외한 내용을 ImportRSAPublicKey 메서드로 직접 로드할 수 있습니다.

string pemText = File.ReadAllText("test_rsa.pub.pkcs1");

var publicKeyBase64 = pemText
    .Replace("-----BEGIN RSA PUBLIC KEY-----", "")
    .Replace("-----END RSA PUBLIC KEY-----", "").Trim();

byte[] publicKeyBytes = Convert.FromBase64String(publicKeyBase64);

using (var rsa = new RSACryptoServiceProvider())
{
    rsa.ImportRSAPublicKey(publicKeyBytes, out _);
    Console.WriteLine(rsa.ToXmlString(false));
}




그렇다면 PEM 포맷으로 변경 없이 곧바로 읽어들일 수 있는 방법은 없을까요? 검색해 보면, 다음과 같은 글이 나옵니다.

Converting OpenSSH public keys
; https://blog.oddbit.com/post/2011-05-08-converting-openssh-public-keys/

위의 글에 따르면 SSH 공개키 파일 포맷은 다음과 같이 정의돼 있습니다.

  [The key type]  [A chunk of PEM-encoded data] [A comment]
예)ssh-rsa        AAAAB3...[생략]...cQ==         testusr@TESTPC

그다음, "[A chunk of PEM-encoded data]"의 포맷은 "key type"이 "ssh-rsa"인 경우 (length, data) 조합의 연속이라고 합니다. 3가지 값이 연속돼 있는데요, 이 형식은 다음과 같습니다.

[4바이트 길이(big endian)] [algorithm name, ("ssh-rsa" | "ssh-dsa")]
[4바이트 길이(big endian)] [RSA exponent (ASN.1)] 
[4바이트 길이(big endian)] [RSA modulus (ASN.1)]

따라서 C#으로는 이런 식으로 코드를 만들 수 있는데요,

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

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

        int exponentLength = reader.ReadInt32BE();
        exponent = reader.ReadBytes(exponentLength);

        int modulusLength = reader.ReadInt32BE();
        modulus = reader.ReadMPINT(modulusLength);
    }

    return (algorithmName, exponent, modulus);
}

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);
    }

    // RSA 공개키 등의 modulus 값에 0x00 선행 바이트가 있는 이유(ASN.1 인코딩)
    // https://www.sysnet.pe.kr/2/0/13740
    public static byte[] ReadMPINT(this BinaryReader reader, int length)
    {
        byte leadByte = reader.ReadByte();

        if (leadByte == 0)
        {
            return reader.ReadBytes(length);
        }

        reader.BaseStream.Seek(-1, SeekOrigin.Current);
        return reader.ReadBytes(length);
    }
}

그런데, 실제로 해보면 algorithmName, exponent 값은 제대로 읽어들이지만 modulus의 값이 틀립니다. 정상적인 경우와 비교해 보면, exponent를 읽어들인 후에 1바이트를 건너 뛰어야 한다는 차이가 있는데요,

int algorithmLength = reader.ReadInt32BE();
algorithmName = Encoding.ASCII.GetString(reader.ReadBytes(algorithmLength));

int exponentLength = reader.ReadInt32BE();
exponent = reader.ReadBytes(exponentLength);

reader.ReadByte();

int modulusLength = reader.ReadInt32BE();
modulus = reader.ReadBytes(modulusLength);

관련해서 검색해 보면 이런 글이 나옵니다.

Converting an RSA Public Key into a RFC 4716 Public Key with Bouncy Castle
; https://stackoverflow.com/questions/15457710/converting-an-rsa-public-key-into-a-rfc-4716-public-key-with-bouncy-castle

즉, exponent 다음에 1바이트 0x00이 추가된다고 합니다. 이게 4바이트 정렬 때문이라 그런 건지, (글에서처럼 puttygen을 포함해) ssh-keygen와 같은 도구에서만 그런 건지는 잘 모르겠습니다. (혹시 이와 관련해 아시는 분은 덧글 부탁드립니다. ^^)

참고로, "Converting OpenSSH public keys" 글에서 제시한 파이썬 예제 코드에도 딱히 1바이트를 건너 뛰어야 하는 코드가 없습니다.

import sys
import base64
import struct

# get the second field from the public key file.
keydata = base64.b64decode(
  open('key.pub').read().split(None)[1])

parts = []
while keydata:
    # read the length of the data
    dlen = struct.unpack('>I', keydata[:4])[0]

    # read in <length> bytes
    data, keydata = keydata[4:dlen+4], keydata[4+dlen:]

    parts.append(data)

아무튼,
저렇게 해서 로드하면 정상적으로 코드가 잘 동작합니다.

string text = File.ReadAllText("test_rsa.pub");

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

    string keyType = sshKeys[0];
    string comment = sshKeys[2];
    byte[] bytesEncoded = Convert.FromBase64String(sshKeys[1]);

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

    (string algorithmName, byte[] exponent, byte[] modulus) = DecodeSSHPublicKey(bytesEncoded);
}

그리고 이렇게 구한 공개키 정보로부터 RSACryptoServiceProvider까지 초기화가 가능합니다.

RSAParameters rsaParam = new RSAParameters();

rsaParam.Exponent = new byte[exponent.Length];
rsaParam.Modulus = new byte[modulus.Length];

Array.Copy(exponent, rsaParam.Exponent, exponent.Length);
Array.Copy(modulus, rsaParam.Modulus, modulus.Length);

RSACryptoServiceProvider rsa = new RSACryptoServiceProvider();
rsa.ImportParameters(rsaParam);

Console.WriteLine(rsa.ToXmlString(false));

/* 출력 결과:
<RSAKeyValue><Modulus>r5zRO5L0MgiySIt4tYWcFZizYXlG/BNaHN0feGC3JzipYrG7l8d4YkhozjzsfVHnX/z6e928fDrbv/Vbna4mfddHD08ww0esyT31GB8g4pbcCd83CqZKBus+opSCVbl9QpvEvavqcJFTpu4o/hUohfEf1sZW5TzGc3U0cgmDAojidCo2V1PGrN4hmnZ0DGQ/llJ+rf4D1CnyLwsNbgCD2OSg6qfESXnjHiScOeUQpJIx879YIWL0AzAqqVl0Pjk+Z9Q0BKWx9ffeFGASYaSQv1vefk59qakz5nHaGz+/p5dwjlcMQe+7pPR8ZH6/ZXQcaGYhB3IYZtag1AqfNkBhWyJrn0NwNsGHUCMkUxw9dYK6Ne44/uVO2nubM3kUM5YekVmsVLz3nxb7OpyoQlzhOck+UrOUKDAY4r8lFnXDD1tl9eXQdEQEdNxwNRpsk5zI9s1SYf+6dISUddVORHDlUPevUX8j3+xr2WZYTEmmDXS4sIODtX/BY0ucJPju7K4poJ0nvivTM9qqKwqLGKeJ2av3XbrsuelCto+tPSW117X0UTdJNevyXg0Lz+0tT7980LcA78gcRkZWe1CoZAAtUdB0Ro6lsKrVTky7aSjQ+uBDRl0mEnXXrBfUggStncEY8oUi7r++HZKR8xmzQJNIE1tZpwoSTeqxyKwbQgOKxUU=</Modulus><Exponent>AQAB</Exponent></RSAKeyValue>
*/




해본 김에, fingerprint 값도 구해볼까요? ^^ 우선 ssh-keygen을 이용해 보면 공개키 파일의 fingerprint는 다음과 같은 명령어로 구할 수 있습니다.

// What is a SSH key fingerprint and how is it generated?
// ; https://superuser.com/questions/421997/what-is-a-ssh-key-fingerprint-and-how-is-it-generated

c:\temp> ssh-keygen -l -f "test_rsa.pub"
4096 SHA256:FfM9wFpzJqN5shtFtt+rk/0ggZbkGthsbsoZz52YiwQ testusr@TESTPC (RSA)

SHA256 해시를 했다는 것에서 대충 다음과 같은 코드를 작성할 수 있습니다.

string text = File.ReadAllText("test_rsa.pub");

string[] sshKeys = text.Split(' ');
string comment = sshKeys[2];
byte[] bytesEncoded = Convert.FromBase64String(sshKeys[1]);

// ...생략... (DecodeSSHPublicKey, RSACryptoServiceProvider 초기화)

string signature = Convert.ToBase64String(SHA256.Create().ComputeHash(bytesEncoded));
Console.WriteLine($"Fingerprint(sha256): {rsa.KeySize} SHA256:{signature.Replace("=", "")} {comment}");

/* 출력 결과:
Fingerprint(sha256): 4096 SHA256:FfM9wFpzJqN5shtFtt+rk/0ggZbkGthsbsoZz52YiwQ testusr@TESTPC
*/

출력 결과가 동일합니다. ^^ 또한, 예전처럼 md5 방식의 해시를 구하는 것도 가능한데요,

c:\temp> ssh-keygen -l -E md5 -f "test_rsa.pub"
4096 MD5:2e:fc:70:c6:6c:09:91:94:c2:4b:50:f5:75:47:ba:5c testusr@TESTPC (RSA)

C# 코드로는 이렇게 작성할 수 있습니다.

signature = BitConverter.ToString(MD5.Create().ComputeHash(bytesEncoded));
Console.WriteLine($"Fingerprint(md5): MD5:{rsa.KeySize} {signature.ToLower().Replace('-', ':')} {comment}");

/* 출력 결과:
Fingerprint(md5): 4096 MD5:2e:fc:70:c6:6c:09:91:94:c2:4b:50:f5:75:47:ba:5c testusr@TESTPC
*/




참고로, 자신의 코드가 제대로 동작하는지 확인하는 용도로 ^^ (비록 유료지만) 아래의 라이브러리가 쓸 만할 것입니다.

SshKey C# Reference Documentation
; https://www.chilkatsoft.com/refdoc/csSshKeyRef.html

Chilkat 라이브러리로 이번 글의 코드도 대충 다음과 같이 작성할 수 있습니다.

string text = File.ReadAllText("test_rsa.pub");
var sshObj = new Chilkat.SshKey();
sshObj.FromOpenSshPublicKey(text);

string xmlKey = sshObj.ToXml();
Console.WriteLine(xmlKey);
Console.WriteLine($"Fingerprint: {sshObj.GenFingerprint()}");

/* 출력 결과:
<RSAPublicKey><Modulus>r5zRO5L0MgiySIt4tYWcFZizYXlG/BNaHN0feGC3JzipYrG7l8d4YkhozjzsfVHnX/z6e928fDrbv/Vbna4mfddHD08ww0esyT31GB8g4pbcCd83CqZKBus+opSCVbl9QpvEvavqcJFTpu4o/hUohfEf1sZW5TzGc3U0cgmDAojidCo2V1PGrN4hmnZ0DGQ/llJ+rf4D1CnyLwsNbgCD2OSg6qfESXnjHiScOeUQpJIx879YIWL0AzAqqVl0Pjk+Z9Q0BKWx9ffeFGASYaSQv1vefk59qakz5nHaGz+/p5dwjlcMQe+7pPR8ZH6/ZXQcaGYhB3IYZtag1AqfNkBhWyJrn0NwNsGHUCMkUxw9dYK6Ne44/uVO2nubM3kUM5YekVmsVLz3nxb7OpyoQlzhOck+UrOUKDAY4r8lFnXDD1tl9eXQdEQEdNxwNRpsk5zI9s1SYf+6dISUddVORHDlUPevUX8j3+xr2WZYTEmmDXS4sIODtX/BY0ucJPju7K4poJ0nvivTM9qqKwqLGKeJ2av3XbrsuelCto+tPSW117X0UTdJNevyXg0Lz+0tT7980LcA78gcRkZWe1CoZAAtUdB0Ro6lsKrVTky7aSjQ+uBDRl0mEnXXrBfUggStncEY8oUi7r++HZKR8xmzQJNIE1tZpwoSTeqxyKwbQgOKxUU=</Modulus><Exponent>AQAB</Exponent></RSAPublicKey>
Fingerprint: ssh-rsa 4096 2e:fc:70:c6:6c:09:91:94:c2:4b:50:f5:75:47:ba:5c
*/

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




마지막으로 표준 관련 이야기를 해볼까요? ^^ "Converting OpenSSH public keys" 글에 따르면 ssh-keygen이 생성한 공개키 파일은 RFC 4253을 따른다고 합니다.

아래의 글에서 좀 더 자세하게 설명하는 글이 나오는데요,

OpenSSH public key file format?
; https://superuser.com/questions/1477472/openssh-public-key-file-format

일단, RFC 4251 문서("The Secure Shell (SSH) Protocol Architecture")상으로는 파일 포맷에 대한 규정이 없다고 합니다. 단지, 직렬화는 필요할 텐데, 재미있는 것은 SSH 프로토콜 협상에서 개인키가 오고 가는 것은 없으므로 그에 대한 직렬화도 스펙에는 없다고 합니다.

단지, 공개키라면 OpenSSH에서 파일에 저장하기 위해 RFC 4253 포맷을 많이 수용했다는 건데요, 파일 저장의 포맷이 규정된 것이 없으므로 클라이언트들이 개별 포맷을 사용하는 듯합니다. 그중 OpenSSH의 경우 내부적으로 OpenSSL을 사용하기 때문에 결국 OpenSSL의 공개키 파일 포맷이 곧 OpenSSH의 포맷이 됩니다.

별도로 "The SSH Public Key Format"으로 RFC 4716이 정의돼 있다고 하는데,

The Secure Shell (SSH) Public Key File Format
; https://datatracker.ietf.org/doc/html/rfc4716

그다지 일반적으로 차용되고 있지는 않다고 합니다. 이 문서 때문인지는 모르겠지만, 종종 Q&A에 보면 ssh-keygen으로 생성한 공개키에 대해 RFC 4716 포맷을 언급하는 글이 있기도 합니다. (혹시 표준 관련해서 잘 알고 계신 분은 덧글 부탁드립니다. ^^)




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

[연관 글]






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

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

비밀번호

댓글 작성자
 



2024-10-18 08시23분
What's new in .NET 9: Cryptography improvements
; https://blog.elmah.io/whats-new-in-net-9-cryptography-improvements/

* KMAC (KECCAK Message Authentication Code)

* KMAC has minimum requirements for your system
  - Linux with OpenSSL 3.0 or later.
  - Windows system of Windows 11 Build 26016 or later.
정성태

1  2  3  4  5  6  7  8  9  10  11  12  13  14  [15]  ...
NoWriterDateCnt.TitleFile(s)
13566정성태2/27/20249542오류 유형: 897. Windows 7 SDK 설치 시 ".NET Development" 옵션이 비활성으로 선택이 안 되는 경우
13565정성태2/23/20248637닷넷: 2219. .NET CLR2 보안 모델에서의 개별 System.Security.Permissions 제어
13564정성태2/22/202410152Windows: 259. Hyper-V Generation 1 유형의 VM을 Generation 2 유형으로 바꾸는 방법
13563정성태2/21/20249645디버깅 기술: 196. windbg - async/await 비동기인 경우 메모리 덤프 분석의 어려움
13562정성태2/21/20249344오류 유형: 896. ASP.NET - .NET Framework 기본 예제에서 System.Web에 대한 System.IO.FileNotFoundException 예외 발생
13561정성태2/20/202410145닷넷: 2218. C# - (예를 들어, Socket) 비동기 I/O에 대한 await 호출 시 CancellationToken을 이용한 취소파일 다운로드1
13560정성태2/19/202410196디버깅 기술: 195. windbg 분석 사례 - Semaphore 잠금으로 인한 Hang 현상 (닷넷)
13559정성태2/19/202411010오류 유형: 895. ASP.NET - System.Security.SecurityException: 'Requested registry access is not allowed.'
13558정성태2/18/202410182닷넷: 2217. C# - 최댓값이 1인 SemaphoreSlim 보다 Mutex 또는 lock(obj)를 선택하는 것이 나은 이유
13557정성태2/18/20249057Windows: 258. Task Scheduler의 Author 속성 값을 변경하는 방법
13556정성태2/17/20249431Windows: 257. Windows - Symbolic (hard/soft) Link 및 Junction 차이점
13555정성태2/15/20249644닷넷: 2216. C# - SemaphoreSlim 사용 시 주의점
13554정성태2/15/20249071VS.NET IDE: 189. Visual Studio - 닷넷 소스코드 디컴파일 찾기가 안 될 때
13553정성태2/14/20248607닷넷: 2215. windbg - thin/fat lock 없이 동작하는 Monitor.Wait + Pulse
13552정성태2/13/20248730닷넷: 2214. windbg - Monitor.Enter의 thin lock과 fat lock
13551정성태2/12/20249818닷넷: 2213. ASP.NET/Core 웹 응용 프로그램 - 2차 스레드의 예외로 인한 비정상 종료
13550정성태2/11/202410264Windows: 256. C# - Server socket이 닫히면 Accept 시켰던 자식 소켓이 닫힐까요?
13549정성태2/3/202411235개발 환경 구성: 706. C# - 컨테이너에서 실행하기 위한 (소켓) 콘솔 프로젝트 구성
13548정성태2/1/202410802개발 환경 구성: 705. "Docker Desktop for Windows" - ASP.NET Core 응용 프로그램의 소켓 주소 바인딩(IPv4/IPv6 loopback, Any)
13547정성태1/31/202410453개발 환경 구성: 704. Visual Studio - .NET 8 프로젝트부터 dockerfile에 추가된 "USER app" 설정
13546정성태1/30/202410143Windows: 255. (디버거의 영향 등으로) 대상 프로세스가 멈추면 Socket KeepAlive로 연결이 끊길까요?
13545정성태1/30/20249683닷넷: 2212. ASP.NET Core - 우선순위에 따른 HTTP/HTTPS 호스트:포트 바인딩 방법
13544정성태1/30/20249340오류 유형: 894. Microsoft.Data.SqlClient - Could not load file or assembly 'System.Security.Permissions, ...'
13543정성태1/30/20249764Windows: 254. Windows - 기본 사용 중인 5357 포트 비활성화는 방법
13542정성태1/30/20248841오류 유형: 893. Visual Studio - Web Application을 실행하지 못하는 IISExpress - 두 번째 이야기
13541정성태1/29/20249601VS.NET IDE: 188. launchSettings.json의 useSSL 옵션
1  2  3  4  5  6  7  8  9  10  11  12  13  14  [15]  ...