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

(시리즈 글이 3개 있습니다.)
스크립트: 23. 파이썬 - WSGI를 만족하는 최소한의 구현 코드 및 PyCharm에서의 디버깅 방법
; https://www.sysnet.pe.kr/2/0/12794

스크립트: 64. 파이썬 - ASGI를 만족하는 최소한의 구현 코드
; https://www.sysnet.pe.kr/2/0/13633

스크립트: 65. 파이썬 - asgi 버전(2, 3)에 따라 달라지는 uvicorn 호스팅
; https://www.sysnet.pe.kr/2/0/13642




파이썬 - asgi 버전(2, 3)에 따라 달라지는 uvicorn 호스팅

지난 글에서 ASGI 버전 3을 만족하는 유형의 asgi 예제를 봤는데요,

# test_main.py

async def simple_app_v3(scope, receive, send):
    event = await receive()

    await send({
        'type': 'http.response.start',
        'status': 200,
        'headers': [
            [b'content-type', b'text/plain'],
        ]
    })

    await send({"type": "http.response.body", "body": b'Hello, world!'})


application = simple_app_v3

그에 앞서 나온 (문서에 Legacy Application으로 표기된) 버전 2는 scope와 receive/send가 분리된 형식으로 돼 있습니다. 이에 대한 예제를 보면,

jordaneremieff/asgi-examples
; https://github.com/jordaneremieff/asgi-examples

def simple_app_v2(scope):
    async def asgi(receive, send):
        await send(
            {
                "type": "http.response.start",
                "status": 200,
                "headers": [[b"content-type", b"text/plain"]],
            }
        )
        await send({"type": "http.response.body", "body": b"Hello, world!"})

    return asgi


application = simple_app_v2


scope을 받는 함수가 내부적으로 send/receive를 정의한 함수를 반환하는 형태로 구성하고 있는데요, 2가지(asgi v2, v3) 모두 uvicorn으로는 이렇게 실행할 수 있습니다.

uvicorn test_main:application --port 18090




부가적으로 uvicorn은 명령행에서 --interface 인자를 통해 asgi 버전을 지정할 수 있습니다. 그래서 simple_app_v2와 simple_app_v3에 따라 각각 다음과 같이 인자를 명시할 수도 있습니다.

[asgi v2]
uvicorn test_main:simple_app_v2 --port 18090 --interface asgi2

[asgi v3]
uvicorn test_main:simple_app_v3 --port 18090 --interface asgi3

만약 지정하지 않았다면 "auto"인데 자동으로 판단해 주는 것입니다. 그렇다면 uvicorn은 어떻게 v2, v3 인터페이스를 판단하는 걸까요? 혹시 def 정의에 붙는 async 유무에 따라 결정할까요?

실제로 이에 대한 테스트를 async 가변 인자에 내부적으로는 v3 규칙을 따르는 함수를 만들고,

async def simple_app_any(*args, **kwargs):
    scope = args[0]
    receive = args[1]
    send = args[2]

    event = await receive() // 내부 코드는 version 3 스펙을 따름

    await send({
        'type': 'http.response.start',
        'status': 200,
        'headers': [
            [b'content-type', b'text/plain'],
        ]
    })

    await send({"type": "http.response.body", "body": b'Hello, world!'})

이것을 uvicorn의 v2로 실행하면 당연히 오류가 발생할 것입니다.

$ uvicorn test_main:simple_app_any --port 18090 --interface asgi2
...[생략]...

Traceback (most recent call last):
  File "/home/kevin/.local/lib/python3.8/site-packages/uvicorn/protocols/http/httptools_impl.py", line 435, in run_asgi
    result = await app(  # type: ignore[func-returns-value]
  File "/home/kevin/.local/lib/python3.8/site-packages/uvicorn/middleware/proxy_headers.py", line 80, in __call__
    return await self.app(scope, receive, send)
  File "/home/kevin/.local/lib/python3.8/site-packages/uvicorn/middleware/asgi2.py", line 20, in __call__
    instance = self.app(scope)
TypeError: 'coroutine' object is not callable)

왜냐하면, v2 스펙상 self.app은 동기 함수로 약속돼 있기 때문에 async 함수로는 오류가 발생하는 것이 맞습니다. (단지, "TypeError: 'coroutine' object is not callable" 오류는 맞지 않는데요, 이에 대해서는 마지막에 설명하겠습니다.)

재미있는 것은 그렇다고 해서 v3 스펙으로 호출해도 여전히 오류가 발생한다는 점입니다.

$ uvicorn test_main:simple_app_any --port 18090 --interface asgi3
...[생략]...

Traceback (most recent call last):
  File "/home/kevin/.local/lib/python3.8/site-packages/uvicorn/protocols/http/httptools_impl.py", line 435, in run_asgi
    result = await app(  # type: ignore[func-returns-value]
  File "/home/kevin/.local/lib/python3.8/site-packages/uvicorn/middleware/proxy_headers.py", line 80, in __call__
    return await self.app(scope, receive, send)
TypeError: 'coroutine' object is not callable)

대신 이번에는 "await self.app(scope, receive, send)"로 비동기 호출을 했다는 것만 다릅니다. 하지만 그런데도 오류가 발생하는 것은 self.app이 (coroutine function이 아니라) coroutine으로 보관돼 있기 때문입니다. 아니... 왜 그런 것일까요? ^^




우선, 이 문제를 해석하려면 Coroutine과 Coroutine Function의 차이를 알아야 합니다. 이에 대해서는 다음의 글에 자세하게 설명하니 생략하고,

Coroutines vs Coroutine Functions in Python
; https://medium.com/geekculture/coroutines-vs-coroutine-functions-in-python-a3957c1a0f9b

따라서, asgi3 규약으로 호출했는데도 uvicorn은 "async def simple_app_any" 함수를 Coroutine Function으로 구분하지 않고 v2의 경우처럼 일반 함수로 인식해 그것에 대해 한 번 더 호출한 함수를 self.app에 보관해 두고 있었다고 추측할 수 있습니다. 결국 self.app에는 coroutine이 들어가게 되고 그것을 await 호출을 했으니 저런 오류가 발생한 것입니다.

그렇다면, 가변 인자를 갖는 함수로 정의된 경우라면 uvicorn이 한번 호출한다는 것을 가정하고 이렇게 만들어 주면 됩니다.

# uvicorn test_main:simple_app_any --port 18090 --interface asgi3


def simple_app_any_wrap():

    async def _handler(*args, **kwargs):
        scope = args[0]
        receive = args[1]
        send = args[2]
        await simple_app_any(scope, receive, send)

    return _handler


async def simple_app_any(*args, **kwargs):
    # ...[생략]...

뭐랄까, uvicorn은 asgi3 인자로 명시했지만 어찌 되었든 v2/v3를 판별하는 코드를 (아마도) scope, receive, send 3개의 인자를 갖고 있는지 여부로 판별하는 듯합니다. 만약 3개의 인자를 갖고 있다면 처음 한 번의 호출을 하지 않고, 그렇지 않다면 호출을 무조건 해버리는 것입니다.

실제로 위의 simple_app_any_wrap 함수에 임의로 3개의 인자를 받게 만들면,

def simple_app_any_wrap(arg1, arg2, arg3):
    # ...[생략]...

이제는 최초 한 번 호출을 하지 않으므로 이런 예외가 발생합니다.

Traceback (most recent call last):
  File "/home/kevin/.local/lib/python3.8/site-packages/uvicorn/protocols/http/httptools_impl.py", line 435, in run_asgi
    result = await app(  # type: ignore[func-returns-value]
  File "/home/kevin/.local/lib/python3.8/site-packages/uvicorn/middleware/proxy_headers.py", line 80, in __call__
    return await self.app(scope, receive, send)
TypeError: object function can't be used in 'await' expression

정리하면, asgi v3인 경우에는,

  • 인자를 아예 받지 않는 동기 함수에서, 한 번 호출되는 것을 가정해 3개의 인자(또는 가변 인자)를 받아 처리하는 함수를 반환하거나,
  • 인자를 3개 받아 처리하는 async 함수를 정의하거나,

반드시 저 2개 함수 정의 중 하나여야 합니다. 즉, 가변 함수를 직접 받는 함수를 정의할 수는 없습니다.




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







[최초 등록일: ]
[최종 수정일: 6/11/2024]

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

비밀번호

댓글 작성자
 




... 106  107  108  109  110  111  112  113  114  115  116  117  118  [119]  120  ...
NoWriterDateCnt.TitleFile(s)
10949정성태4/28/201619866.NET Framework: 575. SharedDomain과 JIT 컴파일파일 다운로드1
10948정성태4/28/201623812.NET Framework: 574. .NET - 눈으로 확인하는 SharedDomain의 동작 방식 [3]파일 다운로드1
10947정성태4/27/201621675.NET Framework: 573. .NET CLR4 보안 모델 - 4. CLR4 보안 모델에서의 조건부 APTCA 역할파일 다운로드1
10946정성태4/26/201624508VS.NET IDE: 106. Visual Studio 2015 확장 - INI 파일을 위한 사용자 정의 포맷 기능 (Syntax Highlighting)파일 다운로드1
10945정성태4/26/201618247오류 유형: 327. VSIX 프로젝트 빌드 시 The "VsTemplatePaths" task could not be loaded from the assembly 오류 발생
10944정성태4/22/201619499디버깅 기술: 80. windbg - 풀 덤프 파일로부터 텍스트 파일의 내용을 찾는 방법
10943정성태4/22/201624354디버깅 기술: 79. windbg - 풀 덤프 파일로부터 .NET DLL을 추출/저장하는 방법 [1]
10942정성태4/19/201619661디버깅 기술: 78. windbg 사례 - .NET 예외가 발생한 시점의 오류 분석 [1]
10941정성태4/19/201619569오류 유형: 326. Error MSB8020 - The build tools for v120_xp (Platform Toolset = 'v120_xp') cannot be found.
10940정성태4/18/201622828Windows: 116. 프로세스 풀 덤프 시간을 줄여 주는 Process Reflection [3]
10939정성태4/18/201623867.NET Framework: 572. .NET APM 비동기 호출의 Begin...과 End... 조합 [3]파일 다운로드1
10938정성태4/13/201623435오류 유형: 325. 파일 삭제 시 오류 - Error 0x80070091: The directory is not empty.
10937정성태4/13/201631652Windows: 115. UEFI 모드로 윈도우 10 설치 가능한 USB 디스크 만드는 방법
10936정성태4/8/201642329Windows: 114. 삼성 센스 크로노스 7 노트북의 운영체제를 USB 디스크로 새로 설치하는 방법 [3]
10935정성태4/7/201626636웹: 32. Edge에서 Google Docs 문서 편집 시 한영 전환키가 동작 안하는 문제
10934정성태4/5/201625373디버깅 기술: 77. windbg의 콜스택 함수 인자를 쉽게 확인하는 방법 [1]
10933정성태4/5/201630975.NET Framework: 571. C# - 스레드 선호도(Thread Affinity) 지정하는 방법 [8]파일 다운로드1
10932정성태4/4/201623273VC++: 96. C/C++ 식 평가 - printf("%d %d %d\n", a, a++, a);
10931정성태3/31/201623548개발 환경 구성: 283. Hyper-V 내에 구성한 Active Directory 환경의 시간 구성 방법 [3]
10930정성태3/30/201621506.NET Framework: 570. .NET 4.5부터 추가된 CLR Profiler의 실행 시 Rejit 기능
10929정성태3/29/201631608.NET Framework: 569. ServicePointManager.DefaultConnectionLimit의 역할파일 다운로드1
10928정성태3/28/201637318.NET Framework: 568. ODP.NET의 완전한 닷넷 버전 Oracle ODP.NET, Managed Driver [2]파일 다운로드1
10927정성태3/25/201626541.NET Framework: 567. System.Net.ServicePointManager의 DefaultConnectionLimit 속성 설명
10926정성태3/24/201626069.NET Framework: 566. openssl의 PKCS#1 PEM 개인키 파일을 .NET RSACryptoServiceProvider에서 사용하는 방법 [10]파일 다운로드1
10925정성태3/24/201620378.NET Framework: 565. C# - Rabin-Miller 소수 생성 방법을 이용하여 RSACryptoServiceProvider의 개인키를 직접 채워보자 - 두 번째 이야기파일 다운로드1
10924정성태3/22/201621013오류 유형: 324. Visual Studio에서 Azure 클라우드 서비스 생성 시 Failed to initialize the PowerShell host 에러 발생
... 106  107  108  109  110  111  112  113  114  115  116  117  118  [119]  120  ...