Microsoft MVP성태의 닷넷 이야기
닷넷: 2192. C# - 특정 실행 파일이 있는지 확인하는 방법 (Linux) [링크 복사], [링크+제목 복사],
조회: 2167
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

C# - 특정 실행 파일이 있는지 확인하는 방법 (Linux)

윈도우와는 달리, 리눅스의 경우 배포본도 다양하지만 컨테이너 환경으로 오면서 더더욱 간소화된 이미지까지 겹쳐 당연히 있을 것 같은 파일들이 없는 경우가 많습니다.

혹시 이런 경우, 뭔가 해당 파일의 유무를 판단할 만한 좋은 근거가 있을까요? ^^; 제가 리알못이라... 가령 systemctl이 있다고 가정했을 때 Ubuntu 20.04에서는 /usr/bin/systemctl에 위치하고 있으므로 이 경로에 대한 판정을 하면 될 것 같은데, 아니면 다른 위치에도 설치되는 경우가 있는 지도 잘 모르겠습니다. (혹시 아시는 분은 덧글 부탁드립니다. ^^)

만약 다양한 경로에 있는 것이 가능하다면, 실행 파일의 유무를 단순하게 다음과 같이 실행해 보고 예외 체크를 하는 식으로 접근할 수 있습니다.

static void Main(string[] args)
{
    Console.WriteLine(HasExecutable("systemctl")); // True
    Console.WriteLine(HasExecutable("systemctl2")); // False
}

static bool HasExecutable(string executable)
{
    ProcessStartInfo psi = new ProcessStartInfo();
    psi.FileName = executable;
    psi.RedirectStandardOutput = true;

    try
    {
        using (Process? process = Process.Start(psi))
        {
            return true;
        }
    }
    catch { }

    return false;
}

단지 기분나쁜 점이 있다면 디버깅 시에 예외가 하나 발생한다는 정도인데요,

Exception thrown: 'System.ComponentModel.Win32Exception' in System.Diagnostics.Process.dll

이를 우회하고 싶다면 which를 이용하는 것도 괜찮을 것 같습니다.

static bool HasExecutable(string executable)
{
    ProcessStartInfo psi = new ProcessStartInfo();
    psi.FileName = "which";
    psi.Arguments = executable;
    psi.RedirectStandardOutput = true;

    using (Process? process = Process.Start(psi))
    {
        if (process == null)
        {
            return false;
        }

        process.WaitForExit();
        return process.ExitCode == 0;
    }
}

하지만 이것도 단점이 있을 듯한데요, 어쩌면 어느 컨테이너인가는 which가 없을 수도 있기 때문입니다. 그렇다면 저 2가지 방법을 혼합하는 것이 그나마 가장 나은 선택일 것 같습니다.

using System.Diagnostics;

namespace ConsoleApp1;

internal class Program
{
    static void Main(string[] args)
    {
        Console.WriteLine(HasExecutable("systemctl"));
        Console.WriteLine(HasExecutable("systemctl2"));
    }

#if DEBUG
    static bool HasExecutable(string executable)
    {
        // ...[생략: which를 이용한 체크]...
    }
#else
    static bool HasExecutable(string executable)
    {
        // ...[생략: 곧바로 실행]...
    }
#endif
}

아니면, 윈도우처럼 Path 환경변수를 돌면서 실행 파일이 있는지 검색하는 것도 좋을 듯하고. (혹시 좀 더 괜찮은 방법을 알고 계신 분이 있다면 덧글 부탁드립니다. ^^)




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







[최초 등록일: ]
[최종 수정일: 1/3/2024]

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)
13494정성태12/20/20232709Linux: 64. Linux 응용 프로그램의 (C++) so 의존성 줄이기(ReleaseMinDependency) - 두 번째 이야기
13493정성태12/19/20232903닷넷: 2185. C# - object를 QueryString으로 직렬화하는 방법
13492정성태12/19/20232557개발 환경 구성: 699. WSL에 nopCommerce 예제 구성
13491정성태12/19/20232355Linux: 63. 리눅스 - 다중 그룹 또는 사용자를 리소스에 권한 부여
13490정성태12/19/20232495개발 환경 구성: 698. Golang - GLIBC 의존을 없애는 정적 빌드 방법
13489정성태12/19/20232266개발 환경 구성: 697. GoLand에서 ldflags 지정 방법
13488정성태12/18/20232234오류 유형: 884. HTTP 500.0 - 명령행에서 실행한 ASP.NET Core 응용 프로그램을 실행하는 방법
13487정성태12/16/20232521개발 환경 구성: 696. C# - 리눅스용 AOT 빌드를 docker에서 수행 [1]
13486정성태12/15/20232328개발 환경 구성: 695. Nuget config 파일에 값 설정/삭제 방법
13485정성태12/15/20232194오류 유형: 883. dotnet build/restore - error : Root element is missing
13484정성태12/14/20232323개발 환경 구성: 694. Windows 디렉터리 경로를 WSL의 /mnt 포맷으로 구하는 방법
13483정성태12/14/20232474닷넷: 2184. C# - 하나의 resource 파일을 여러 프로그램에서 (AOT 시에도) 사용하는 방법파일 다운로드1
13482정성태12/13/20233153닷넷: 2183. C# - eFriend Expert OCX 예제를 .NET Core/5+ Console App에서 사용하는 방법 [2]파일 다운로드1
13481정성태12/13/20232469개발 환경 구성: 693. msbuild - .NET Core/5+ 프로젝트에서 resgen을 이용한 리소스 파일 생성 방법파일 다운로드1
13480정성태12/12/20232872개발 환경 구성: 692. Windows WSL 2 + Chrome 웹 브라우저 설치
13479정성태12/11/20232506개발 환경 구성: 691. WSL 2 (Ubuntu) + nginx 환경 설정
13477정성태12/8/20232743닷넷: 2182. C# - .NET 7부터 추가된 Int128, UInt128 [1]파일 다운로드1
13476정성태12/8/20232501닷넷: 2181. C# - .NET 8 JsonStringEnumConverter의 AOT를 위한 개선파일 다운로드1
13475정성태12/7/20232578닷넷: 2180. .NET 8 - 함수 포인터에 대한 Reflection 정보 조회파일 다운로드1
13474정성태12/6/20232397개발 환경 구성: 690. 닷넷 코어/5+ 버전의 ilasm/ildasm 실행 파일 구하는 방법 - 두 번째 이야기
13473정성태12/5/20232682닷넷: 2179. C# - 값 형식(Blittable)을 메모리 복사를 이용해 바이트 배열로 직렬화/역직렬화파일 다운로드1
13472정성태12/4/20232336C/C++: 164. Visual C++ - InterlockedCompareExchange128 사용 방법
13471정성태12/4/20232524Copilot - To enable GitHub Copilot, authorize this extension using GitHub's device flow
13470정성태12/2/20232774닷넷: 2178. C# - .NET 8부터 COM Interop에 대한 자동 소스 코드 생성 도입파일 다운로드1
13469정성태12/1/20232623닷넷: 2177. C# - (Interop DLL 없이) CoClass를 이용한 COM 개체 생성 방법파일 다운로드1
13468정성태12/1/20232424닷넷: 2176. C# - .NET Core/5+부터 달라진 RCW(Runtime Callable Wrapper) 대응 방식파일 다운로드1
1  2  3  4  5  [6]  7  8  9  10  11  12  13  14  15  ...