Microsoft MVP성태의 닷넷 이야기
닷넷: 2361. C# - Linux 환경의 readlink 호출 [링크 복사], [링크+제목 복사],
조회: 712
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일
 

C# - Linux 환경의 readlink 호출

리눅스 쪽은 API보다는 파일 시스템을 기반으로 한 정보들이 많습니다. 심지어 파일 자체의 내용뿐만 아니라 링크에서도 정보를 구하게 되는 경우가 종종 있는데요, 예를 들어 프로세스의 pid 네임스페이스에 대한 것도,

pid 네임스페이스 구성으로 본 WSL 2 배포본의 계층 관계
; https://www.sysnet.pe.kr/2/0/13772

링크로 연결이 됩니다.

$ ls -l /proc/4419/ns/pid
lrwxrwxrwx 1 testusr testusr 0 Oct 17 14:46 /proc/4419/ns/pid -> 'pid:[4026532257]'

리눅스 시스템에서 이 값을 코드로 구하기 위해서는 readlink 또는 realpath를 사용할 수 있는데,

readlink(1) - Linux man page
; https://linux.die.net/man/1/readlink

readlink(2) — Linux man page
; https://linux.die.net/man/2/readlink

realpath(1) - Linux man page
; https://linux.die.net/man/1/realpath

realpath(3) - Linux man page
; https://linux.die.net/man/3/realpath

$ realpath /proc/self/ns/pid
/proc/683623/ns/pid:[4026531836]

$ readlink -f /proc/self/ns/pid
/proc/683762/ns/pid:[4026531836]

$ readlink /proc/self/ns/pid
pid:[4026531836]

그렇다면 C#으로는 어떻게 구할 수 있을까요? 아쉽게도 .NET 5까지의 기본 라이브러리에는 이를 위한 배려가 없었기 때문에 libc를 interop하는 식으로 구해야만 했습니다.

public class Program
{
    [DllImport("libc.so.6", CharSet = CharSet.Ansi)]
    internal static extern int readlink(string path, byte[] buf, ulong bufsiz);

    public static void Main(string[] args)
    {
        byte[] buf = new byte[1024];
        readlink("/proc/self/ns/pid", buf, 1024);
        Console.WriteLine($"{Encoding.ASCII.GetString(buf)}"); // 출력 결과: pid:[4026531836]
    }
}

// 또는, shell을 경유해 readlink(1) 명령어를 Process.Start로 실행한 결과로 받아도 됩니다.

그러다가, 마이크로소프트도 이에 대한 필요성을 인식했는지 .NET 6부터는 File과 Directory 타입에 각각 ResolveLinkTarget 정적 메서드를 추가했고,

File.ResolveLinkTarget(String, Boolean) Method
; https://learn.microsoft.com/en-us/dotnet/api/system.io.file.resolvelinktarget

Directory.ResolveLinkTarget(String, Boolean) Method
; https://learn.microsoft.com/en-us/dotnet/api/system.io.directory.resolvelinktarget

이를 이용해 다음과 같이 readlink/realpath에 해당하는 기능을 호출할 수 있습니다.

string path = "...";
FileSystemInfo? fsi = null;

if (File.Exists(path))
{
    FileAttributes attrs = File.GetAttributes(path);
    Console.WriteLine(attrs);
    if ((attrs & FileAttributes.ReparsePoint) == FileAttributes.ReparsePoint)
    {
        fsi = File.ResolveLinkTarget(path, true);
    }
}
else if (Directory.Exists(path))
{
    FileAttributes attrs = File.GetAttributes(path);
    Console.WriteLine(attrs);
    if ((attrs & FileAttributes.ReparsePoint) == FileAttributes.ReparsePoint)
    {
        fsi = Directory.ResolveLinkTarget(path, true);
    }
}

if (fsi != null)
{
    Console.WriteLine($"{fsi.FullName}");
    Console.WriteLine($"{fsi.Name}");
}

예를 들어, path == "/proc/self/ns/pid"와 같은 파일인 경우 다음과 같은 출력을 얻고,

ReadOnly, ReparsePoint
/proc/self/ns/pid:[4026531836]
pid:[4026531836]

"/bin"과 같은 디렉터리라면 이렇게 나옵니다.

ReadOnly, Directory, ReparsePoint
/usr/bin
bin

참고로, ResolveLinkTarget 메서드는 윈도우 파일 시스템의 링크에도 사용할 수 있습니다.




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







[최초 등록일: ]
[최종 수정일: 9/7/2025]

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

비밀번호

댓글 작성자
 




... 91  92  93  94  95  96  97  98  99  100  101  102  [103]  104  105  ...
NoWriterDateCnt.TitleFile(s)
11447정성태1/20/201824774.NET Framework: 730. dotnet user-secrets 명령어 [2]파일 다운로드1
11446정성태1/20/201824418.NET Framework: 729. windbg로 살펴보는 GC heap의 Segment 구조 [2]파일 다운로드1
11445정성태1/20/201822785.NET Framework: 728. windbg - 눈으로 확인하는 Workstation GC / Server GC
11444정성태1/19/201822257VS.NET IDE: 125. Visual Studio에서 Selenium WebDriver를 이용한 웹 브라우저 단위 테스트 구성파일 다운로드1
11443정성태1/18/201824858VC++: 124. libuv 모듈 살펴 보기
11442정성태1/18/201820691개발 환경 구성: 353. ASP.NET Core 프로젝트의 "Enable unmanaged code debugging" 옵션 켜는 방법
11441정성태1/18/201818824오류 유형: 447. ASP.NET Core 배포 오류 - Ensure that restore has run and that you have included '...' in the TargetFrameworks for your project.
11440정성태1/17/201822253.NET Framework: 727. ASP.NET의 HttpContext.Current 구현에 대응하는 ASP.NET Core의 IHttpContextAccessor/HttpContextAccessor 사용법파일 다운로드1
11439정성태1/17/201828749기타: 69. C# - CPU 100% 부하 주는 프로그램파일 다운로드1
11438정성태1/17/201821813오류 유형: 446. Error CS0234 The type or namespace name 'ITuple' does not exist in the namespace
11437정성태1/17/201821581VS.NET IDE: 124. Platform Toolset 설정에 따른 Visual C++의 헤더 파일 기본 디렉터리
11436정성태1/16/201824176개발 환경 구성: 352. ASP.NET Core (EXE) 프로세스가 IIS에서 호스팅되는 방법 - ASP.NET Core Module(AspNetCoreModule) [4]
11435정성태1/16/201825700개발 환경 구성: 351. OWIN 웹 서버(EXE)를 IIS에서 호스팅하는 방법 - HttpPlatformHandler (Reverse Proxy)파일 다운로드2
11434정성태1/15/201826942개발 환경 구성: 350. 사용자 정의 웹 서버(EXE)를 IIS에서 호스팅하는 방법 - HttpPlatformHandler (Reverse Proxy)파일 다운로드2
11433정성태1/15/201824824개발 환경 구성: 349. dotnet ef 명령어 사용을 위한 준비
11432정성태1/11/201832249.NET Framework: 726. WPF + Direct2D + SharpDX 출력 C# 예제파일 다운로드2
11431정성태1/11/201827880.NET Framework: 725. C# - 동기 방식이면서 비동기 메서드(awaitable)처럼 구현한 사례 [9]
11430정성태1/10/201831315.NET Framework: 724. WPF + Direct2D 출력 C# 예제 [2]파일 다운로드1
11429정성태1/9/201821598개발 환경 구성: 348. ASP.NET Core 2.1 Preview 버전 적용 방법
11428정성태1/6/201825384개발 환경 구성: 347. WinForm 프로젝트를 WPF 프로젝트 유형으로 변경하는 방법파일 다운로드1
11427정성태1/5/201822094오류 유형: 445. vcpkg 빌드 오류 - Starting the CLR failed with HRESULT 80040153
11426정성태1/5/201833172오류 유형: 444. curl로 호출할 때 발생하는 오류 정리
11425정성태1/4/201823804개발 환경 구성: 346. ASP.NET Core Web Application을 IIS에서 호스팅하는 방법 (2)
11424정성태1/4/201823058개발 환경 구성: 345. ASP.NET Core 프로젝트를 명령행에서 빌드하는 방법
11423정성태1/3/201841623VC++: 123. 내가 만든 코드보다 OpenCV의 속도가 월등히 빠른 이유 [8]파일 다운로드2
11422정성태1/2/201831393.NET Framework: 723. C# - OpenCvSharp 사용 시 C/C++을 이용한 속도 향상 (for 루프 연산) [4]파일 다운로드1
... 91  92  93  94  95  96  97  98  99  100  101  102  [103]  104  105  ...