June.

10월 안에는 꼭...

Portfolio

About

에셋 관리 시스템을 만들어보자 (에셋타운 3편 - Cloudflare Workers, R2를 이용한 웹훅 처리와 CDN 캐싱)

2024/03/01
개발
에셋타운

7 min read

에셋 관리 시스템을 만들어보자 (에셋타운 3편 - Cloudflare Workers, R2를 이용한 웹훅 처리와 CDN 캐싱)

이 시리즈는 총 4개의 컨텐츠로 기획되어 있습니다.

  1. 에셋 관리 시스템을 만들어보자 (에셋타운 1편 - 개요)
  2. 에셋 관리 시스템을 만들어보자 (에셋타운 2편 - Sanity를 이용한 어드민)
  3. 에셋 관리 시스템을 만들어보자 (에셋타운 3편 - Cloudflare Workers, R2를 이용한 웹훅 처리와 CDN 캐싱) [현재 글]
  4. 에셋 관리 시스템을 만들어보자 (에셋타운 4편 - Figma Plugin을 이용한 에셋 전시)

해당 프로젝트는 일주일 만에 만들어진 프로젝트로 추후에는 많은 변경이 있을 수도 있습니다.

Sanity의 CDN을 이용하지 않는 이유

Sanity에서 데이터베이스에 에셋들을 올리면 기본적으로 CDN URL이 제공됩니다. 하지만 이전에 1편에서 언급했듯, CDN 한 달 무료 범위는 1,000,000 (백만) 요청까지 무료입니다.

백만까지 무료
백만까지 무료

1,000,000 이라는 숫자가 크게 보일 수 있지만, 단순 조회에 대한 숫자이기 때문에 실제로는 엄청 빠르게 올라갈 수 있습니다. 특히 제가 현재 몸 담고 있는 당근과 같은 트래픽이 많은 곳들에서는 이런 비용이 큰 부담이 될 수 있습니다. 그래서 신경써주지 않으면 과금 폭탄을 맞을 수 있습니다.

그래서 이번 글에서는 에셋에 대한 원본은 Sanity에 저장하고 에셋에 대한 수정이나 삭제나 추가에 대한 이벤트를 받아서 Cloudflare Workers에 전달되고 Cloudflare Workers에서 R2에 캐싱을 하고 CDN URL을 제공하는 방법에 대해 설명하겠습니다.

Sanity Webhook 등록하기

Webhook은 특정 이벤트가 발생했을 때, 지정된 URL로 HTTP 요청을 보내는 것을 말합니다.

위에서 설명되어있듯, Webhook은 특정 이벤트가 발생했을 때, 지정된 URL로 HTTP 요청을 보내는 것을 말합니다.

Sanity에서는 Create, Update, Delete에 대한 이벤트 Webhook을 등록할 수 있습니다.

Sanity에서 Webhook 등록하는 곳
Sanity에서 Webhook 등록하는 곳

Sanity에서는 총 2개의 Webhook을 무료로 등록할 수 있습니다. 이 Webhook을 처리해줄 서버를 만들어야 합니다. 우리는 이 서버를 Cloudflare Workers를 이용해서 만들거구요.

간단하게 Webhook 등록 화면을 살펴볼까요?

Webhook 등록 화면
Webhook 등록 화면

위에서부터 하나씩 살펴보면

  1. Name (Webhook 이름)
  2. Description (Webhook 설명)
  3. URL (Webhook을 처리해줄 서버 URL)
  4. Dataset (어떤 데이터셋에 대한 Webhook인지)
  5. Trigger On (어떤 이벤트에 대한 Webhook인지)
  6. Filter (어떤 필터를 걸어서 Webhook을 처리할지)
  7. Projection (어떤 필드에 대한 Webhook인지)

이렇게 7개 옵션 이외에도 Status, HTTP Method, Headers 등의 옵션을 설정할 수 있습니다. 요기서 저희는 3번부터 7번까지 집중하면 좋은데요, 어떤식으로 웹훅 전략을 가져갈지 생각해보겠습니다.

Webhook 전략

URL은 Cloudflare Workers를 생성하게 되면 기본적인 URL을 제공받을 수 있습니다. 해당 URL을 Webhook URL로 등록하면 됩니다.

총 두 개의 Webhook을 무료로 만들 수 있기 때문에 Create, Update에 대한 이벤트를 동시에 처리하는 Webhook 하나랑 Delete를 처리하는 Webhook 하나를 만들어서 이렇게 두 개로 운영할겁니다.

  • Create, Update에 대한 이벤트를 받아서 처리할 Webhook: https://<your-worker-url>/api/update
  • Delete에 대한 이벤트를 받아서 처리할 Webhook: https://<your-worker-url>/api/delete

그리고 Dataset 같은 경우는 Webhook의 Headers에 담겨져 오는 데이터를 통해서 Develop 환경과 Production 환경에 대한 처리를 따로 해줄 수 있기 때문에 all dataset으로 처리할 것입니다.

Webhook에 Headers에 담겨져 오는 데이터는 아래와 같습니다. (참고)

1{ 2 "conection": "close", 3 "accept-encoding": "gzip", 4 "idempotency-key": "<a unique key>", 5 "content-type": "application/json", 6 "content-length": "<the length of the payload in bytes>", 7 "user-agent": "Sanity.io webhook delivery", 8 "host": "<the endpoint URL host>", 9 "sanity-transaction-id": "<ID of transaction>", 10 "sanity-transaction-time": "<Timestamp of transaction>", 11 "sanity-dataset": "<Name of dataset (also available in projection today as sanity::dataset())>", 12 "sanity-document-id": "<Document ID being notified about>", 13 "sanity-project-id": "<ID of project (also available in projection today as sanity::projectId())>", 14 "sanity-webhook-id": "<ID of webhook>" 15}

그리고 Filter와 Projection 같은 경우는 Webhook을 처리할 때 필요한 데이터를 걸러내거나 필요한 데이터만 가져올 수 있도록 하는 옵션입니다. GROQ 쿼리를 이용하기 때문에 자신의 프로젝트에 맞게 설정하면 됩니다. 저 같은 경우에는 Sanity에서 에셋에 대한 메타데이터 대부분을 들고옵니다. Cloudflare R2에 저장할 때 Custom Metadata 필드에 저장해서 추후에 이용할 수 있도록 합니다.

참고로 저는 Webhook의 Filter와 Projection을 아래와 같이 설정했습니다.

1// Filter 2*[_type in ["lottie", "svg", "gif"]] 3 4// Projection 5{ 6 _id, 7 _type, 8 title, 9 description, 10 lottie -> { 11 url, 12 originalFilename, 13 uploadId, 14 assetId, 15 extension, 16 mimeType, 17 path, 18 _id, 19 _type, 20 _createdAt, 21 _updatedAt 22 }, 23 svg -> { 24 ... 위와 동일 25 }, 26 gif -> { 27 ... 위와 동일 28 } 29}

Cloudflare Workers를 이용한 Webhook 처리

Cloudflare Workers는 Cloudflare에서 제공하는 서버리스 플랫폼입니다. 이를 이용하여 자바스크립트 코드로 HTTP 요청을 처리하는 서버를 만들 수 있습니다.

이제는 Cloudflare Workers를 이용하여 Webhook을 처리하는 서버를 만들어보겠습니다.

Cloudflare Workers는 기본적으로 JavaScript 혹은 TypeScript로 작성할 수 있습니다. 그리고 REST API를 이용하여 HTTP 요청을 처리할 수 있습니다. 그러기 위해서는 itty-router를 이용하여 라우팅을 처리할 수 있습니다.

위에서 설명했듯, 우리는 두 개의 Webhook을 처리할 것이기 때문에 두 개의 Route를 만들어야 합니다.

1// https://github.com/junghyeonsu/asset-town/blob/main/workers/src/index.ts 2import { Router } from "itty-router"; 3 4import { handleOptions } from "./helper"; // CORS 처리를 위한 함수 5import { deleteLottieFromBucket, putLottieFromBucket } from "./api/lottie"; 6 7const router = Router(); 8 9router.put("/api/v1/lottie", putLottieFromBucket); 10router.delete("/api/v1/lottie", deleteLottieFromBucket);

위 코드에서 보이는 것처럼 putLottieFromBucketdeleteLottieFromBucket 함수를 만들어서 각각의 Route에 대한 요청을 처리할 수 있도록 합니다.

현재는 lottie에 대한 요청만 처리할 수 있도록 만들었지만, 다른 에셋에 대한 요청도 처리할 수 있도록 만들 수 있습니다.

Cloudflare는 기본적으로 각 서비스끼리 연동성이 좋아서 R2 버킷을 생성하고 Workers의 설정 파일에 R2 버킷을 연동할 수 있습니다.

1name = "sanity-proxy" 2main = "src/index.js" 3compatibility_date = "2022-07-12" 4 5workers_dev = false 6 7{/* Workers와 R2 버킷 연결 */} 8r2_buckets = [ 9 { binding = "ASSET_TOWN_CDN_BUCKET", bucket_name = "asset-town-cdn-prod"}, 10]

위와 같이 설정을 하고 나면 아래와 같이 코드를 작성할 수 있습니다.

1// 에셋에 대해서 들고오는 타입 2interface SanityAsset { 3 url: string; 4 originalFilename: string; 5 uploadId: string; 6 assetId: string; 7 extension: string; 8 mimeType: string; 9 path: string; 10 _id: string; 11 _type: string; 12 _createdAt: string; 13 _updatedAt: string; 14} 15 16// Projection에 대한 타입 17interface SanityRequest { 18 _id: string; 19 _type: string; 20 title: string; 21 description: string; 22 23 // 에셋들 (Filter를 통해서 가져온 에셋들입니다.) 24 // Filter: *[_type in ["lottie", "svg", "gif"]] 25 lottie: SanityAsset; 26 svg: SanityAsset; 27 gif: SanityAsset; 28} 29 30// src/api/lottie.ts 31 32const SANITY_USER_AGENT = "Sanity.io webhook delivery"; // Webhook을 처리할 때 User-Agent를 체크하기 위한 상수 33 34/** 35 * lottie에 대한 요청을 처리하는 함수 36 * 37 * request는 Webhook을 통해서 들어오는 요청에 대한 정보가 들어있습니다. 38 * env는 Cloudflare Workers의 설정 파일에 있는 R2 버킷에 대한 정보가 들어있습니다. 39 * 자세한 내용은 공식문서를 참고해주세요: https://developers.cloudflare.com/r2/api/workers/workers-api-usage/ 40 */ 41export const putLottieFromBucket = async (request: Request, env: Env): Promise<Response> => { 42 // Webhook을 처리할 때 User-Agent를 체크합니다. 43 if (request.headers.get("user-agent") !== SANITY_USER_AGENT) { 44 return new Response(JSON.stringify({ message: "Unauthorized" }), { 45 headers: { 46 "Content-Type": "application/json", 47 "Access-Control-Allow-Origin": "*", 48 }, 49 status: 401, 50 }); 51 } 52 53 // request 객체에 들어있는 정보를 JSON으로 변환합니다. 54 // 그럼 Webhook 설정에서 설정한 Filter와 Projection에 대한 정보를 가져올 수 있습니다. 55 const asset = await request.json<SanityRequest>(); 56 57 // Webhook을 처리할 때 Dataset에 대한 정보를 가져옵니다. 58 // 환경 분리를 위해서 사용합니다. 59 const dataset = request.headers.get("sanity-dataset"); 60 61 if (asset.lottie) { 62 const lottie = await fetch(asset.lottie.url, { 63 headers: { 64 "Content-Type": "application/json", 65 }, 66 }).then((response) => response.json()); 67 /** 68 * R2는 /로 폴더를 구분합니다. 69 * @example development/lottie/1234/5678.json 70 * @example production/lottie/abcd/efgh.json 71 * 위와 같이 구분할 수 있습니다. 72 * 73 * R2에 저장할 데이터 스키마는 자신이 정하기 나름입니다. 74 */ 75 const lottiePath = `${dataset}/${asset._type}/${asset._id}/${asset.lottie.assetId}.${asset.lottie.extension}`; 76 77 try { 78 // env.{wrangler.toml에 설정한 R2 버킷 이름}으로 R2에 대한 다양한 메서드를 사용할 수 있습니다. 79 await env.ASSET_TOWN_CDN_BUCKET.put(lottiePath, JSON.stringify(lottie), { 80 httpMetadata: { 81 contentType: asset.lottie.mimeType, 82 }, 83 // Custom Metadata에 대한 정보를 넣을 수 있습니다. 84 customMetadata: { 85 ...asset.lottie, 86 }, 87 }); 88 } catch { 89 return new Response(JSON.stringify({ message: "Lottie Put Failed!!!" }), { 90 headers: { 91 "Content-Type": "application/json", 92 "Access-Control-Allow-Origin": "*", 93 }, 94 status: 500, 95 }); 96 } 97 } 98}

Delete에 대한 함수도 위와 같이 비슷하게 작성됩니다. 그리고 현재는 lottie 에셋에 대해서만 처리를 할 수 있지만, 함수를 조금 리팩토링한다면 다른 에셋에 대해서도 처리할 수 있습니다.

위와 같이 Cloudflare Workers를 이용하여 Webhook을 처리할 수 있습니다. 이렇게 하고 이제 Sanity에서 에셋을 업로드하거나 수정하거나 삭제하면 Webhook이 발생하고 Cloudflare Workers에 전달되게 됩니다. 그러면 Cloudflare Workers에서 해당 에셋을 다시 R2에 저장합니다.

R2를 이용한 CDN 캐싱

Cloudflare R2는 Cloudflare에서 제공하는 버킷 서비스입니다. 이 버킷을 이용하여 에셋을 캐싱할 수 있습니다. AWS의 S3와 비슷한 역할을 합니다.

Sanity는 CDN 조회에 대한 비용을 청구하지만, Cloudflare R2는 CDN 조회에 대한 비용을 청구하지 않습니다. 단, 조건은 자신의 커스텀 도메인을 R2에 연결하고 R2를 Public하게 오픈해야 합니다.

그리고 Cloudflare R2에서 제공하는 기본적인 캐싱 정책에 대한 문서에 적혀있는 기본적으로 캐싱이 되는 파일들의 확장자들은 아래와 같습니다.

R2에서 기본적으로 캐시해주는 확장자들
R2에서 기본적으로 캐시해주는 확장자들

이렇게 R2에 커스텀 도메인까지 연결하면 CDN 조회에 대한 비용을 청구하지 않고 유저에게 에셋에 대한 Public URL을 무료로 제공할 수 있습니다.

Sanity에서 제공하는 CDN 조회에 대한 비용을 줄이기 위해 약간의 해키한 방법을 사용했습니다.

Sanity에서 날라오는 Webhook 정보를 통해서 Cloudflare R2에 저장할 때 예측 가능하고 확장 가능한 URL 포맷을 정의하고, 그 포맷에 맞게 저장해야 합니다.

그리고 유저에게 URL을 제공할 때도 R2에 저장된 URL을 제공해야 하기 때문에 Sanity CDN to R2 CDN과 같은 유틸 함수를 생성하거나 매핑을 해야 합니다.

1// utils/url.ts 2 3/** 4 * Sanity CDN URL을 R2 CDN URL로 변환하는 함수 5 * 6 * @param {string} url - Sanity CDN URL 7 * @param {string} dataset - Dataset 이름 8 * @param {string} type - 에셋 타입 9 * @param {string} id - 에셋 ID 10 * @param {string} assetId - 에셋 Asset ID 11 * @param {string} extension - 에셋 확장자 12 * @returns {string} - R2 CDN URL 13 */ 14export const sanityCdnToR2Cdn = (url: string, dataset: string, type: string, id: string, assetId: string, extension: string): string => { 15 const r2CdnUrl = new URL(url); 16 r2CdnUrl.hostname = `{your-r2-custom-domain}`; // R2 커스텀 도메인 17 r2CdnUrl.pathname = `${dataset}/${type}/${id}/${assetId}.${extension}`; 18 return r2CdnUrl.toString(); 19};

마치며

이번 글에서는 에셋타운에서 Cloudflare Workers와 R2를 이용하여 웹훅을 처리하고 CDN 캐싱을 제공하는 방법에 대해 설명했습니다. 다음 글에서는 Figma Plugin을 이용하여 에셋을 전시하는 방법에 대해 설명하겠습니다.

참고

관련 포스트가 4개 있어요.

ESM, CJS, TS를 모두 지원하는 라이브러리를 만들어보고, CLI를 간단하게 만들어보겠습니다.

개발
npm
deploy-simple-npm-library cover image
2024/03/19 (updated)

간단한 NPM 라이브러리 배포해보기 2탄 (CLI 간단한 라이브러리 만들기 & 모든 환경 지원, ESM, CJS, TS)

Figma Plugin을 이용하여 에셋을 전시하는 방법에 대해 알아봅니다.

개발
에셋타운
asset-town-figma-plugin cover image
2024/03/17

에셋 관리 시스템을 만들어보자 (에셋타운 4편 - Figma Plugin을 이용한 에셋 전시)

에셋타운의 어드민 페이지 Sanity를 이용해서 만들어보기

개발
에셋타운
asset-town-admin-with-sanity cover image
2024/02/17

에셋 관리 시스템을 만들어보자 (에셋타운 2편 - 어드민 페이지 with Sanity)

에셋타운이 무엇이고 왜 필요한지 그리고 어떻게 만들어 나갈지에 대한 개요를 설명합니다.

개발
에셋타운
asset-town-introduction cover image
2024/02/03

에셋 관리 시스템을 만들어보자 (에셋타운 1편 - 개요)

profile

정현수.

Currently Managed

Currently not managed

© 2024. junghyeonsu all rights reserved.