Microsoft MVP성태의 닷넷 이야기
닷넷: 2353. C# - Foundry Local을 이용한 gpt-oss-20b 모델 사용 [링크 복사], [링크+제목 복사],
조회: 744
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일

(시리즈 글이 6개 있습니다.)
개발 환경 구성: 748. Windows + Foundry Local - 로컬에서 AI 모델 활용
; https://www.sysnet.pe.kr/2/0/13943

닷넷: 2337. C# - Hugging Face에 공개된 LLM 모델을 Foundry Local에서 사용하는 방법
; https://www.sysnet.pe.kr/2/0/13954

닷넷: 2338. C# / Foundry Local - Phi-4-multimodal 모델을 사용하는 방법
; https://www.sysnet.pe.kr/2/0/13957

닷넷: 2339. C# - Phi-4-multimodal 모델의 GPU 가속 방법 (ORT 사용)
; https://www.sysnet.pe.kr/2/0/13958

닷넷: 2348. C# - 카카오 카나나 모델 + Microsoft.ML.OnnxRuntimeGenAI 예제
; https://www.sysnet.pe.kr/2/0/13976

닷넷: 2353. C# - Foundry Local을 이용한 gpt-oss-20b 모델 사용
; https://www.sysnet.pe.kr/2/0/13992




C# - Foundry Local을 이용한 gpt-oss-20b 모델 사용

오호~~~ 최근 OpenAI에서 GPT OSS 20B 모델을 공개했는데요, Hugging Face에도 올라온 상태입니다.

openai/gpt-oss-20b
; https://huggingface.co/openai/gpt-oss-20b/blob/main/config.json

아쉽게도 "GptOssForCausalLM" 구조라 olive를 이용한 ONNX 포맷으로의 전환이 안 되는 유형이었는데, 마이크로소프트에서 발 빠르게 이것을 Foundry Local에 기본 지원 모델로 포함시켰기 때문에,

Available today: gpt-oss-20B Model on Windows with GPU Acceleration – further pushing the boundaries on the edge
; https://blogs.windows.com/windowsdeveloper/2025/08/05/available-today-gpt-oss-20b-model-on-windows-with-gpu-acceleration-further-pushing-the-boundaries-on-the-edge/

지난 글에서 설명한 방법대로 C#에서도 손쉽게 접근할 수 있습니다.

Windows + Foundry Local - 로컬에서 AI 모델 활용
; https://www.sysnet.pe.kr/2/0/13943




그래도 한번 실습을 해볼까요? ^^ 일단 olive 변환은 할 수 없으니, Foundry Local을 이용해 다음과 같이 다운로드할 수 있습니다.

C:\temp> foundry model download gpt-oss-20B
Downloading gpt-oss-20b-cuda-gpu...
[####################################] 100.00 % [Time remaining: about 0s]        36.9 MB/s
Tips:
- To find model cache location use: foundry cache location
- To find models already downloaded use: foundry cache ls

이후 OpenAI 패키지로 Foundry Local과 연동해 이런 식으로 코딩할 수 있습니다.

using OpenAI;
using OpenAI.Chat;
using System.ClientModel;

namespace ConsoleApp1;

internal class Program
{
    // Install-Package OpenAI 
    static void Main(string[] args)
    {
        string ep = "http://localhost:5273/v1";
        string key = "OPENAI_API_KEY";
        string alias = "gpt-oss-20b-cuda-gpu";

        OpenAIClientOptions options = new OpenAIClientOptions();
        options.Endpoint = new Uri(ep);

        ApiKeyCredential akc = new ApiKeyCredential(key);
        ChatClient client = new(alias, akc, options);

        ChatCompletion completion = client.CompleteChat("하늘이 파란 이유는?'");

        foreach (var message in completion.Content)
        {
            Console.WriteLine($"[{message.Kind}]: {message.Text}");
        }
    }
}

/* 실행 결과:

[Text]: <|channel|>analysis<|message|>The user says: "하늘이 파란 이유는?" in Korean, which translates to "The reason why the sky is blue?" The question is likely about the reason behind Rayleigh scattering, color of sky because of scattering of shorter wavelengths of visible light off atmosphere, etc.

We need to respond. The user didn't give any context besides asking. They just say in Korean: "The reason the sky is blue?" So answer: It's due to Rayleigh scattering causing blue light to be scattered more.

We can provide explanation: solar light: white, but Earth's atmosphere scatters more of blue wavelengths, causing blue sky.

So just answer like: "태양빛이 투과하면서 대기 중 분자와 아주 작은 입자에 의해 산란된 파장에서 가장 짧은 파장이 산란이 가장 잘 일어나므로..." or we can keep simple.

We can also mention the "Huygens–Fresnel principle" or "Mie scattering
*/

참고로, 모델 용량이 11GB 정도여서 그런지 초기 로딩 시간이 꽤 걸리는군요. ^^

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




Foundry Local을 통해 다운로드한 모델의 경우 .\Microsoft\gpt-oss-20b-cuda-gpu\v1 디렉터리에 genai_config.json 파일이 함께 있습니다. 아하... 그렇다면 Microsoft.ML.OnnxRuntimeGenAI 패키지를 이용하는 것도 가능하다는 의미일 텐데요,

using Microsoft.ML.OnnxRuntimeGenAI;
using System.Reflection;
using System.Reflection.Emit;

namespace ConsoleApp2;

internal class Program
{
    // Install-Package Microsoft.ML.OnnxRuntimeGenAI.CUDA

    static void Main(string[] args)
    {
        // cuDNN 필요
        string? path = Environment.GetEnvironmentVariable("PATH");
        path += @";C:\Program Files\NVIDIA\CUDNN\v9.10\bin\12.9";
        Environment.SetEnvironmentVariable("PATH", path);

        string modelPath = @"C:\foundry_cache\Microsoft\gpt-oss-20b-cuda-gpu\v1";

        Console.Write("Loading model from " + modelPath + "...");
        using Model model = new(modelPath);
        Console.Write("Done\n");
        using Tokenizer tokenizer = new(model);
        using TokenizerStream tokenizerStream = tokenizer.CreateStream();

        while (true)
        {
            Console.Write("User:");

            string prompt = "<|im_start|>user\n" +
                            Console.ReadLine() +
                            "<|im_end|>\n<|im_start|>assistant\n";
            var sequences = tokenizer.Encode(prompt);

            using GeneratorParams gParams = new GeneratorParams(model);
            gParams.SetSearchOption("max_length", 2400);
            using Generator generator = new(model, gParams);
            generator.AppendTokenSequences(sequences);

            Console.Out.Write("\nAI:");
            while (!generator.IsDone())
            {
                generator.GenerateNextToken();
                var token = generator.GetSequence(0)[^1];
                Console.Out.Write(tokenizerStream.Decode(token));
                Console.Out.Flush();
            }
            Console.WriteLine();
        }
    }
}

아쉽게도 실행해 보면 이런 오류가 발생합니다.

Loading model from C:\foundry_cache\Microsoft\gpt-oss-20b-cuda-gpu\v1...Unhandled exception. Microsoft.ML.OnnxRuntimeGenAI.OnnxRuntimeGenAIException: Load model from E:\foundry_cache\Microsoft\gpt-oss-20b-cuda-gpu\v1\model.onnx failed:This is an invalid model. In Node, ("/model/layers.0/attn/GroupQueryAttention", GroupQueryAttention, "com.microsoft", -1) : ("/model/layers.0/attn/qkv_proj/Add/output_0": tensor(float16),"","","past_key_values.0.key": tensor(float16),"past_key_values.0.value": tensor(float16),"/model/attn_mask_reformat/attn_mask_subgraph/Sub/Cast/output_0": tensor(int32),"/model/attn_mask_reformat/attn_mask_subgraph/Gather/Cast/output_0": tensor(int32),"cos_cache": tensor(float16),"sin_cache": tensor(float16),"","","model.layers.0.attn.sinks": tensor(float16),) -> ("/model/layers.0/attn/GroupQueryAttention/output_0": tensor(float16),"present.0.key": tensor(float16),"present.0.value": tensor(float16),) , Error Node(/model/layers.0/attn/GroupQueryAttention) with schema(com.microsoft::GroupQueryAttention:1) has input size 12 not in range [min=7, max=11].
at Microsoft.ML.OnnxRuntimeGenAI.Model..ctor(String modelPath)
at ConsoleApp2.Program.Main(String[] args)


음... 아마도 Microsoft.ML.OnnxRuntimeGenAI 패키지가 업데이트되기를 기다려야 할 것 같습니다. ^^ (기록을 보니까 불과 5일 전에 0.9.0 업데이트가 되었는데 그 버전이 안 됩니다.)




혹시나 Foundry Local에서 gpt-oss-20B 모델을 찾지 못한다고 나오면?

C:\foundry_cache> foundry model run gpt-oss-20B
Exception: Model gpt-oss-20B not found

지난 버전의 Foundry Local을 사용하고 있는 경우인데요, 최신 버전으로 업데이트하면 됩니다.




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







[최초 등록일: ]
[최종 수정일: 8/12/2025]

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

비밀번호

댓글 작성자
 




... 31  32  33  34  35  36  37  [38]  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
13068정성태5/24/202220853.NET Framework: 2018. C# - 일정 크기를 할당하는 동안 GC를 (가능한) 멈추는 방법 [1]파일 다운로드1
13067정성태5/23/202217384Windows: 206. Outlook - 1년 이상 지난 메일이 기본적으로 안 보이는 문제
13066정성태5/23/202218177Windows: 205. Windows 11 - Windows + S(또는 Q)로 뜨는 작업 표시줄의 검색 바가 동작하지 않는 경우
13065정성태5/20/202219714.NET Framework: 2017. C# - Windows I/O Ring 소개 [2]파일 다운로드1
13064정성태5/18/202218959.NET Framework: 2016. C# - JIT 컴파일러의 인라인 메서드 처리 유무
13063정성태5/18/202219386.NET Framework: 2015. C# - 인라인 메서드(inline methods)
13062정성태5/17/202219344.NET Framework: 2014. C# - async/await 그리고 스레드 (4) 비동기 I/O 재현 [1]파일 다운로드1
13061정성태5/16/202219122.NET Framework: 2013. C# - FILE_FLAG_OVERLAPPED가 적용된 파일의 읽기/쓰기 시 Position 관리파일 다운로드1
13060정성태5/15/202222753.NET Framework: 2012. C# - async/await 그리고 스레드 (3) Task.Delay 재현파일 다운로드1
13059정성태5/14/202220611.NET Framework: 2011. C# - CLR ThreadPool의 I/O 스레드에 작업을 맡기는 방법 [1]파일 다운로드1
13058정성태5/13/202220391.NET Framework: 2010. C# - ThreadPool.SetMaxThreads 사용법 [1]
13057정성태5/12/202221984오류 유형: 812. 파이썬 - ImportError: cannot import name ...
13056정성태5/12/202217423.NET Framework: 2009. C# - async/await 그리고 스레드 (2) MyTask의 호출 흐름 [2]파일 다운로드1
13055정성태5/11/202222762.NET Framework: 2008. C# - async/await 그리고 스레드 (1) MyTask로 재현 [11]파일 다운로드1
13054정성태5/11/202219174.NET Framework: 2007. C# - 10진수 숫자를 담은 문자열을 숫자로 변환하는 방법 [11]파일 다운로드1
13053정성태5/10/202217917.NET Framework: 2006. C# - GC.KeepAlive 메서드의 역할
13052정성태5/9/202219198.NET Framework: 2005. C# - 생성한 참조 개체가 언제 GC의 정리 대상이 될까요?
13051정성태5/8/202218486.NET Framework: 2004. C# XingAPI - ACF 검색 결과로 구한 CSV 파일을 통해 퀀트 종목 찾기파일 다운로드1
13050정성태5/6/202218062.NET Framework: 2003. C# - COM 개체의 이벤트 핸들러에서 발생하는 예외에 대한 CLR의 특별 대우파일 다운로드1
13049정성태5/6/202214374오류 유형: 811. GoLand - Error: Cannot find package
13048정성태5/6/202217423오류 유형: 810. "ASUS TUF GAMING B550M-PLUS (WI-FI)" 모델에서 블루투스 장치가 인식이 안 되는 문제
13047정성태5/6/202216818오류 유형: 809. Speech Recognition could not start
13046정성태5/5/202217647.NET Framework: 2002. C# XingAPI - ACF 파일을 이용한 퀀트 종목 찾기(t1857)
13045정성태5/5/202219184.NET Framework: 2001. C# XingAPI - 주식 종목에 따른 PBR, PER, ROE 구하는 방법(t3341 예제)
13044정성태5/4/202217453오류 유형: 808. error : clang++ exited with code 127
13043정성태5/3/202215403오류 유형: 807. C# - 닷넷 응용 프로그램에서 Informix DB 사용 시 오류 메시지 정리
... 31  32  33  34  35  36  37  [38]  39  40  41  42  43  44  45  ...