안녕하세요? Justkode 입니다. 이번 시간에는 AWS Lambda에 대해서 알아 보고, AWS Lambda 함수를 생성 해 보고, 이를 AWS API Gateway를 통해 연결 하여 사용 해 보는 시간을 가져 보도록 하겠습니다.
AWS Lambda에 대해서 간단하게 설명 하고 넘어 가자면, **PaaS (Platform As A Service)**의 일종으로, 우리가 AWS Lambda에 우리가 구현한 API Code를 등록 하고, 이에 대한 컴퓨팅 비용을 사용한 만큼만 지불 하는 서비스 입니다.
우리가 따로 운영체제, 런타임 등의 관리를 할 필요 없이 사용자가 Python, Node.js 등의 언어로 만들어진 Lambda Handler Function을 구현 하여 이를 AWS Lambda에 등록하고, 이를 AWS API Gateway에 연결 하면, 외부에서 해당 API를 이용 할 수 있습니다.
AWS Lambda
먼저, 사용 하기 전에, 일단 Lambda 함수를 만들어 보는 시간을 가져 보도록 하겠습니다. AWS Lambda Console에 먼저 접속을 해 주세요. 그 다음, 우측 상단에 있는 함수 생성을 클릭 해 주세요.
AWS Lambda Console 접속 화면
그러면, 이렇게 함수를 생성 하는 창으로 이동하게 됩니다. 첫 번째로, Lambda Handler의 이름을 지정 해 보겠습니다. 함수 이름에 원하는 함수명을 입력 하고, 원하는 **런타임(언어)**를 선택 해 줍니다.
함수 생성 화면
그 밑에는 함수에 대한 접근 권한 (VPC), 코드 서명을 설정 할 수 있습니다. 이 강의에서는 다루지 않습니다. 세팅이 완료 되셨다면, 우측 하단의 함수 생성을 클릭 하시면 됩니다.
함수 생성 화면
이렇게 함수가 생성된 것을 확인 할 수 있습니다. 그럼 한 번, 스크롤을 내려서 코드를 입력 하는 곳을 확인 해 볼까요?
함수 생성 완료
다음과 같이 코드를 입력 할 수 있는 창이 있는 것을 확인 할 수 있습니다. 여기서 왜 lambda_handler
같은 함수가 나왔는지, 엔트리 함수는 무엇인지 궁금 해 하실 수 있습니다. 한 번 더 스크롤을 내려 볼까요?
코드 입력
그럼 다음과 같은 정보들을 확인 할 수 있는데요, 여기서 런타임 설정 - 핸들러를 확인 하시게 되면, 엔트리 함수가 무엇인지 확인 할 수 있습니다. 일단 다시 돌아가서 엔트리 (혹은 핸들러) 함수에 대한 코드 분석을 해 보도록 하겠습니다.
코드 입력
핸들러 함수는 다음과 같은 파라미터를 갖습니다.
event
: 이벤트와 관련한 정보를 담고 있습니다. event
객체를 통해서, json형식으로 온 body에 접근 할 수 있습니다. 이 객체는 일반적으로 Python dict
유형입니다. 또한 list
, str
, int
, float
또는 NoneType
유형이 될 수 있습니다.추가로, 이 예제에서는 Lambda Proxy를 사용 합니다. (여담: 이 Post 쓰면서 이 친구 때문에 하루 삽질함 ^^, 제발 AWS 형님들 도큐먼트좀 이쁘게 써주세요. 검색하면 최상단에 나오게 했으면서 가독성이 진짜 쓰레기야. 우리 한테 "???: 야, 이래도 써? 징하다 징해" 하는 것도 아니고 어?)
Lambda Proxy를 사용 하면, Event 객체로 들어오는 데이터는 매~우 많습니다. 대충 어떤 값이 나오는지는 Test 옆 ▼ 버튼 - Configure test event를 눌러 주세요, 그 다음, 테스트 이벤트 구성 - 이벤트 템플릿 - Amazon API Gateway AWS Proxy를 클릭 해보면 다음과 같이 파라미터를 확인 해 볼 수 있습니다.
body
: Body로 들어온 String 값 입니다.httpMethod
: 어떤 HTTP Method 를 사용 하여 요청하였는지에 대한 정보를 담고 있습니다.isBase64Encoded
: Base64로 인코딩 되었는지 알려 줍니다.queryStringParameters
: Query String을 파싱하여 어떤 파라미터가 들어 왔는지 알려 줍니다.multiValueQueryStringParameters
: Query String을 파싱하여 어떤 파라미터가 들어 왔는지 알려 줍니다. 중복된 키를 가진, 여러 가지 Value에 대해서도 출력합니다.headers
: HTTP Header 정보를 알려 줍니다.multiValueHeaders
: HTTP Header 정보를 알려 줍니다. 중복된 키를 가진, 여러 가지 Value에 대해서도 출력합니다.
클릭 해 보면,
엄청 많음.
context
: 컨텍스트 정보를 담고 있습니다. 이 객체는 호출, 함수 및 런타임 환경에 관한 정보를 제공하는 메서드 및 속성들을 제공합니다.한 번, 간단한 계산 연산을 해 주는 코드를 구현 해 볼까요? 아래에 있는 코드는, 연산자, 피연산자 정보를 가지고 있는 해당 json
을 body
로 입력 받아, 계산 명령을 실행 하는 코드입니다. 일단 새로운 테스트 이벤트 생성을 클릭 후 테스트 이벤트 구성 - 이벤트 템플릿 - Amazon API Gateway AWS Proxy를 선택하여 템플릿을 가져온 다음, Event Json을 복사해서 넣어 주세요. 다음, 이벤트 이름을 add 라고 한 후에 저장 해 주세요.
{
"body": null,
"resource": "/{proxy+}",
"path": "/path/to/resource",
"httpMethod": "GET",
"isBase64Encoded": true,
"queryStringParameters": {
"a": "1",
"b": "2",
"operator": "+"
},
"multiValueQueryStringParameters": {
"a": [
"1"
],
"b": [
"2"
],
"operator": [
"+"
]
},
...
}
다음과 같이 넣어 주세요.
한 번 코드를 구현 해 보겠습니다. 방금 저장한 테스트 이벤트에 작성한 json 객체가 그대로 lambda_handler
의 event
파라미터로 들어 갑니다. 이를 알아 두고 코딩을 하면 되겠습니다.
또한, Lambda 프록시 통합 옵션을 나중에 사용하는 API Gateway에서 사용하게 된다면, lambda_handler
는 다음과 같은 반환 형식을 유지 하여야 합니다.
{
"isBase64Encoded" : "boolean",
"statusCode": "number",
"headers": { ... },
"body": "JSON string"
}
자, 이에 맞게 한 번 프로그래밍 해 볼까요? a
와 b
로 들어 온 값에 대해 operator
연산자에 해당하는 연산을 하는 API를 만들어 보도록 하겠습니다.
import json
def lambda_handler(event, context):
# Key Check
param = event['queryStringParameters']
print(param)
if any([e not in param for e in ('operator', 'a', 'b')]):
return {
'statusCode': 400,
'body': json.dumps({"message": "key error"})
}
operator = param['operator']
try:
a = int(param['a'])
b = int(param['b'])
except Exception:
return {
'statusCode': 400,
'body': json.dumps({
"message": "value error"
})
}
if operator == '+' or operator == ' ': # query string에서 +는 띄어쓰기로 변환됨
return {
'statusCode': 200,
'body': json.dumps({
"result": a + b
})
}
elif operator == '-':
return {
'statusCode': 200,
'body': json.dumps({
"result": a - b
})
}
elif operator == '/':
return {
'statusCode': 200,
'body': json.dumps({
"result": a / b
})
}
elif operator == '*':
return {
'statusCode': 200,
'body': json.dumps({
"result": a * b
})
}
else:
return {
'statusCode': 400,
'body': json.dumps({"message": "operator error"})
}
코드를 완성 했다면, 이제 저장을 하고 배포를 할 시간 입니다. Ctrl + S 입력 후, Deploy를 클릭 해 줍니다. 그러면, Changes deployed 가 뜨면 성공입니다!
Test 옆 Deploy 클릭
그 다음, 우리가 만든 함수를 테스트 해봐야죠! Test를 누르면, 다음과 같이 결과가 나오게 됩니다.
Test 결과
Test를 누른 후, 모니터링 - 로그를 클릭하면, Lambda 함수를 실행 하면서 출력된 로그들을 열람 할 수 있습니다.
최근의 LogStream을 확인 해 보면, 이렇게 print로 출력 했던 파라미터들을 확인 할 수 있다.
다음은 API Gateway Console로 이동 합니다. 우리가 만든 AWS Lambda 함수를 API Gateway에 등록 하기 위해서 입니다. 일단 스크롤을 좀 내려서, REST API - 구축을 클릭 해 보죠.
REST API - 구축 클릭
그 다음, 프로토콜 선택에서는 REST, 새 API 생성 에서는 새 API, 설정에서는 API 이름을 입력 해 주고, 설명이 필요하다면, 설명도 작성 합니다. 다 입력 하셨다면, API 생성을 클릭 합니다.
REST API - 구축 클릭
그러면, 이제 우리가 만든 API와 관련한 대시보드가 나오게 됩니다. 일단 좌측의 메뉴에서 API - 리소스를 클릭하고, 리소스 - 작업 - 리소스 생성을 클릭 합니다. 하위 리소스를 설정 해야 하기 때문이죠. 하위 리소스를 분리 하여 생성 함으로써, 우리는 하나의 API에 대해 여러 가지 함수를 붙여 넣을 수 있습니다. 비슷한 기능을 하는 함수끼리 묶을 수가 있다는 뜻이죠. (ex: /user/login (로그인), /user/modify (유저 정보 수정 등))
리소스 - 작업 - 리소스 생성 클릭
그 다음에는 우측에 새 하위 리소스를 생성하는 창이 뜨게 됩니다. 리소스 이름과 리소스 경로를 입력 해 주고, CORS 옵션이 필요 하다면, API Gateway CORS 활성화도 체크 해 줍니다. 여기서 리소스 경로는 등록 할 API의 경로를 입력 해 주시면 됩니다.
다 완료하면 리소스 생성 클릭
그 다음, 우리가 만든 함수를 등록 할 차례 입니다. 리소스 - 작업 - 메서드 생성을 클릭 해 주세요. 타입은 ANY를 선택 해 줍니다. 우리는 AWS Lambda Proxy를 사용 하기 때문에, HTTP Method 정보를 가져 올 수 있기 때문이죠.
리소스 - 작업 - 메서드 생성 클릭
ANY 클릭후 옆에 체크 무늬 클릭
그러면 이렇게 메서드를 설정하는 창이 뜨게 되는데요. 통합 유형 - Lambda 함수를 선택 해 주시고, Lambda 프록시 통합 사용 - 체크, Lambda 함수는 방금 만드신 AWS Lambda 함수의 이름을 입력 해 주세요, 리전에 맞게 하시면 됩니다.
다음과 같이 설정 하시고, 저장
마지막으로, 이렇게 우리가 만든 API를 스테이지에 올려 실제로 배포 할 시간입니다. 리소스 - 작업 - API 배포를 클릭 해 주세요.
리소스 - 작업 - API 배포 클릭
그럼 배포할 스테이지를 선택하라는 창이 뜹니다. 그러면 **배포 스테이지 - [새 스테이지]**를 클릭, 스테이지 이름을 입력 합니다. 그 다음, 스테이지 설명 및 배포 설명을 기호에 맞게 작성 후, 저장을 눌러 주세요.
입력 후 저장
자! 그러면 이렇게 우리가 만든 API를 스테이지에 등록 완료 하였습니다. 위에 있는 URL 호출에 있는 엔드포인트를 복사 하여 한번 테스트 해 보겠습니다.
입력 후 저장
성공적으로 작동 되는 것을 볼 수가 있습니다!
반환도 잘 되고, Status Code도 잘 적용 되는 모습.
이렇게 AWS Lambda를 사용해 보는 시간을 가져 보았습니다. 이 글에서는 다루지 않았지만, 트리거 기능을 이용하여, S3 이미지가 들어오면 자동으로 이미지 리사이징을 해 주는 등의 기능 또한 있으니, 트리거에 대해서 따로 공부 해 보시는 것도 추천드립니다! 다음 시간에는 AWS RDS, AWS DynamoDB에 대해서 공부 해 보는 시간을 가져 보도록 하겠습니다.