<p style="text-align:justify;">며칠 전, 회사 팀원 분을 통해 구글에서 진행한 온라인 컨퍼런스인 <a href="https://web.dev/live/">web.dev LIVE</a>를 알게 되었습니다. 슬쩍 둘러보니 재밌어 보이는 주제가 몇 개 보여서, 주말에 시간을 내서 세션을 시청했습니다. 저는 특히 그중에서 <a href="https://youtu.be/vsMJiNtQWvw">Building better in the world of build tools!</a> 를 인상 깊게 봤습니다. 웹팩과 같은 JavaScript 번들링 툴이 왜 필요하고, 어떤 종류가 있으며 어떤 장단점을 갖고 있는지에 대해서 간단하게 요약한 영상이었습니다.</p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">그래서 사실 이 영상을 번역해서 글을 써보고자 했는데, 영상이 설명보다는 개발자들끼리 썰 푸는(?) 것 위주라 정리하기가 쉽지 않았습니다. 그래서 영상 자체 내용보다는 영상에 나온 키워드 위주로 관련 자료들을 리서치했죠.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">필수 개념을 설명하려다 보니 번들러와 모듈 시스템의 역사까지 설명해야 했기에… <i>요거를 어떻게 하면 재미있게 설명할 수 있을까</i> 를 생각하다가 문득 </span><a href="https://namu.wiki/w/%ED%83%95%EC%88%98%EC%9C%A1%EC%9C%BC%EB%A1%9C%20%EB%B3%B8%20%EC%A1%B0%EC%84%A0%EC%8B%9C%EB%8C%80%20%EB%B6%95%EB%8B%B9%EC%9D%98%20%EC%9D%B4%ED%95%B4">붕당정치 짤</a>이 <span style="color:#212529;">생각이 났습니다. 그래서 거기에 맞게 글을 쓰면 재미있을 것 같다는 생각이 들어서 한 번 짤을 만들어 봤습니다.</span></p><p style="text-align:justify;"> </p><figure class="image image_resized" style="width:100%;"><img src="https://yozm.wishket.com/media/news/1261/image001.png"><figcaption><i>약간(?)의 각색이 포함되어 있습니다</i></figcaption></figure><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">고증을 최대한 반영하고 싶긴 했는데 딱 맞아떨어지는 예시가 많이 없어서 골치가 좀 아팠습니다. 어느 정도까지 이해가 가시나요? 아는 만큼 보일 것입니다. 혹시라도 이해가 잘 가지 않는 분들을 위해 지금부터는 모듈과 번들러의 역사에 대한 간략한 설명을 해볼까 합니다. 그렇다고 해서 조선시대 역사까지 알 필요는 없습니다. 저도 그냥 대충 끼워 맞춘 거라서…</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">아무튼 이 글을 통해 <strong>JavaScript 모듈 시스템에 대한 이해가 필요한 분</strong>들, 혹은 <strong>JavaScript 번들러 생태계에 대한 이해가 필요한 분</strong>들께 도움이 되기를 바랍니다.</span></p><hr><h3 style="text-align:justify;"><span style="color:#212529;"><strong>모듈 시스템</strong></span></h3><p style="text-align:justify;"><span style="color:#212529;">최초의 JavaScript는 아주 간단한 모듈 시스템만을 제공했습니다. 바로 HTML에서 JavaScript 원본 소스를 제공하고, 브라우저에서 이것을 순서대로 로드하는 방식이었죠.</span></p><p style="text-align:justify;"> </p><pre><code class="language-plaintext"><html> <script src="/src/foo.js"></script> <script src="/src/bar.js"></script> <script src="/src/baz.js"></script> <script src="/src/qux.js"></script> <script src="/src/quux.js"></script> </html></code></pre><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">이 방법은 그럴듯해 보였지만, 스크립트를 로드한 전역 컨텍스트에서 각 모듈 간의 충돌이 발생한다는 문제가 있었죠. </span><span style="color:#E83E8C;"><code>foo.js</code></span><span style="color:#212529;">에서 선언한 변수와 같은 이름을 가진 변수가 </span><span style="color:#E83E8C;"><code>bar.js</code></span><span style="color:#212529;">에 선언되어 있다면, 나중에 호출된 </span><span style="color:#E83E8C;"><code>bar.js</code></span><span style="color:#212529;">의 변수로 재정의가 되면서 </span><span style="color:#E83E8C;"><code>foo.js</code></span><span style="color:#212529;">의 파일이 제대로 동작하지 않게 되었습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">즉 <strong>모듈 간의 스코프가 구분이 되지 않아 다른 파일을 오염시키는 경우</strong>가 발생했다는 이야기죠. 그래서 개발자들은 다른 모듈과 변수 이름이 겹치지 않도록 모듈 로드 순서를 지정하는 데에 불필요하게 많은 시간을 쏟곤 했습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">시간이 흐르고, 웹이 단순히 문서를 보여주는 역할에 그치지 않고 JavaScript를 활용한 애플리케이션 플랫폼에 가까워지게 되다 보니 상황은 더욱 복잡해졌습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">특히 2008년은 브라우저 외부에서도 JavaScript를 실행시킬 수 있는 Google의 V8 엔진이 공개된 해였기 때문에, 이를 이용해 서버 사이드에서 JavaScript를 활용하자는 아이디어가 막 제시되던 시기였죠. 그러다 보니 모듈화에 대한 필요성이 더욱 부각되었고, 결국 2009년을 기점으로 JavaScript의 모듈을 표준화하기 위한 움직임이 본격적으로 시작되었습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">바로 <strong>CommonJS</strong>와 <strong>AMD(Asynchronous Module Definition)</strong>의 등장입니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"> </p><h3 style="text-align:justify;"><span style="color:#212529;"><strong>CommonJS</strong></span></h3><figure class="image image_resized" style="width:80%;"><img src="https://yozm.wishket.com/media/news/1261/image002.png"></figure><p style="text-align:justify;"><a href="http://www.commonjs.org/">CommonJS</a><span style="color:#212529;">는 <i>Common</i>이라는 이름에서 나와있듯, JavaScript를 브라우저뿐만 아니라 서버 사이드 애플리케이션이나 데스크톱 애플리케이션 등 좀 더 범용적인 용도로 사용하기 위한 모듈 시스템을 만들기 위해 조직된 자발적인 그룹입니다. 그래서 해당 조직에서 제안한 방법을 CommonJS, 또는 CJS라고 부르죠.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">사용 문법은 아래와 같습니다.</span></p><pre><code class="language-plaintext">// CommonJS // 모듈 정의 module.exports = foo; // 모듈 사용 const foo = require("./foo");</code></pre><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">CommonJS는 애초에 브라우저 외의 환경에서도 동작하는 범용적인 JavaScript를 위한 모듈 시스템이기 때문에, <strong>모든 디펜던시가 로컬 디스크에 존재해서 필요한 모듈을 바로 사용할 수 있는 환경을 전제로 합니다. 따라서 동기적(synchronously)으로 모듈을 호출하는 방식을 선택</strong>했습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#E83E8C;"><code>require</code></span><span style="color:#212529;">과 </span><span style="color:#E83E8C;"><code>exports</code></span><span style="color:#212529;"> 키워드를 활용한 직관적이고 간단한 문법이 눈에 띕니다. 하지만 비동기 방식보다 느리고, 트리 쉐이킹(tree shaking, 임포트 되었지만 실제로는 사용되지 않은 코드를 분석하고 삭제하는 코드 최적화 기술)이 어려운 데다 순환 참조에 취약하다는 등의 단점 또한 명확했죠. 하지만 초창기 JavaScript 생태계의 발전에 있어서 모듈 시스템은 반드시 필요했기 때문에, JavaScript 런타임인 Node.js는 CommonJS 방식의 명세를 채택하고 구현했습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">아시다시피 Node.js의 선택은 옳았고 NPM의 성장과 함께 JavaScript 생태계의 규모는 폭발적으로 증가했습니다. 이 때문에 CommonJS의 모듈 시스템의 사용 빈도 역시 높아졌지만, 처음부터 비동기 로드를 고려하지 않은 설계 때문에 브라우저에서는 CommonJS를 사용할 수 없다는 문제가 있었죠. 즉, CommonJS 모듈 시스템은 Node.js 프로젝트에 있어서는 훌륭한 해결책이 되었지만, 브라우저에게 있어서는 그렇지 않았다는 이야기입니다. 이 때문에 CommonJS 모듈 방식을 브라우저 단에서도 활용하기 위한 최초의 빌드 도구인 </span><a href="http://browserify.org/">Browserify</a><span style="color:#212529;">가 탄생하기에 이릅니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"> </p><h3 style="text-align:justify;"><span style="color:#212529;"><strong>AMD</strong></span></h3><figure class="image image_resized" style="width:80%;"><img src="https://yozm.wishket.com/media/news/1261/image003.jpg"><figcaption><span style="color:#495057;"><i>AMD 방식의 대표적인 모듈 로더인 RequireJS</i></span></figcaption></figure><p style="text-align:justify;"> </p><p style="text-align:justify;"><a href="https://github.com/amdjs/amdjs-api/wiki/AMD">AMD</a><span style="color:#212529;">(Asynchronous Module Definition)는 비동기 상황에서도 JavaScript 모듈을 쓰기 위해 CommonJS에서 함께 논의하다 합의점을 이루지 못하고 독립한 별도의 그룹입니다. 사실 초창기 AMD가 만들어진 이유 역시 브라우저에서의 모듈 실행을 우선적으로 여겼기 때문입니다. 브라우저에서는 필요한 모듈들을 네트워크를 통해 비동기적으로 다운로드하고 나서야 사용할 수 있었던 탓이죠.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">사용 문법은 아래와 같습니다.</span></p><pre><code class="language-plaintext">// AMD // 모듈 정의 define([ 'jquery', 'underscore', // 의존 모듈들을 배열로 나열 ], function ($, _) { // 의존 모듈들은 순서대로 매개변수에 담김 return { // 외부에 노출할 함수들만 반환 }; }); // 모듈 사용 require([ ... // 사용할 모듈 배열로 나열 ], function (...) { // 사용할 모듈들이 순서대로 매개변수에 담김 });</code></pre><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">아무래도 복잡한 문법이 띄긴 하지만, 비동기적으로 모듈을 호출하는 특성 때문에 퍼포먼스 면에서는 CommonJS 보다 나은 성능을 보였습니다. 그리고 브라우저와 서버 사이드에서 모두 호환되는 방식이기도 했죠. AMD 명세로 구현된 대표적인 모듈 로더 라이브러리로는 </span><a href="https://requirejs.org/">RequireJS</a><span style="color:#212529;">가 있습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"> </p><h3 style="text-align:justify;"><span style="color:#212529;"><strong>UMD</strong></span></h3><p style="text-align:justify;"><span style="color:#212529;">CommonJS와 AMD는 서로 지향하는 목적이 달랐기 때문에, 어떤 것이 더 낫다고 이야기하기는 쉽지 않습니다. 하지만 이처럼 통일되지 않은 규격들의 발생은 결국 서로 간의 호환성 문제로 이어졌죠.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">이 때문에 등장한 것이 바로</span> <a href="https://github.com/umdjs/umd">UMD</a>(Universal Module<span style="color:#212529;">Definition) 패턴입니다. <i>Universal</i>이라는 이름에서 알 수 있듯이, CommonJS와 AMD 방식을 모두 호환할 수 있도록 조건문으로 분기하고, 이를 팩토리 패턴으로 구현했죠.</span></p><p style="text-align:justify;"> </p><pre><code class="language-plaintext">(function (root, factory) { if (typeof define === "function" && define.amd) { // AMD 방식 define(["jquery", "underscore"], factory); } else if (typeof exports === "object") { // CommonJS 방식 module.exports = factory(require("jquery"), require("underscore")); } else { root.foo = factory(root.$, root._); } })(this, function ($, _) { // 모듈 정의 var foo = { // ... }; return foo; });</code></pre><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">생김새가 그다지 좋아 보이진 않지만, 이 방법은 CommonJS와 AMD 방식을 모두 호환하면서 </span><span style="color:#E83E8C;"><code>window</code></span><span style="color:#212529;"> 객체를 통해 전역적으로도 접근할 수 있습니다. 임시방편으로라도 두 방법의 코드를 모두 호환할 수 있었기 때문에, 라이브러리를 만들 때 자주 사용되는 패턴이 되었죠.</span></p><figure class="table" style="text-align:justify;"><table><tbody><tr><td style="background-color:hsl(0, 0%, 90%);"><span style="color:#495057;">이후에 설명할 웹팩(Webpack)과 롤업(Rollup) 같은 몇몇 JavaScript 번들러들은 ES6 방식으로 모듈 로드에 실패했을 때, 대안책(fallback)으로 UMD 패턴으로 로드하는 방식을 아직도 사용하고 있다고 합니다.</span></td></tr></tbody></table></figure><p style="text-align:justify;"> </p><p style="text-align:justify;"> </p><h3 style="text-align:justify;"><span style="color:#212529;"><strong>ES6 Module</strong></span></h3><figure class="image image_resized" style="width:100%;"><img src="https://yozm.wishket.com/media/news/1261/image004.jpg"></figure><p style="text-align:justify;"><span style="color:#212529;">하지만 UMD 역시 CommonJS와 AMD의 호환성만을 해결할 뿐이었기 때문에, 모듈 시스템의 부재라는 근본적인 문제를 해결하지는 못했습니다. 때문에 JavaScript 언어 자체에서 모듈 시스템을 지원해야 한다는 필요성이 높아졌죠.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">그래서 2015년에, ES6라고도 불리는 ECMAScript 6 사양에서 드디어 JavaScript의 표준 모듈 시스템이 명세되었습니다. 이를 <strong>ES6 Module(ES Module)</strong>이라고 부르죠.</span></p><pre><code class="language-plaintext">// ES6 import foo from "bar"; export default qux;</code></pre><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">위에서의 진통을 겪고 나서 만들어진 JavaScript의 공식 모듈 시스템인 만큼, 그 과정에서 나타나는 문제점들을 해결하기 위해 많은 노력을 기울였습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">우선 동기/비동기 로드를 모두 지원하고, 문법 역시 간단합니다. 또한 CommonJS와는 다르게 실제 객체/함수를 바인딩하기 때문에 순환 참조 관리도 편합니다. 또한 정적 분석(static analyze, 코드를 실행하지 않더라도 분석이 가능함)이 가능하기 때문에, 트리 쉐이킹 역시 쉽게 가능해졌습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">하지만 비교적 최근에 정의된 문법이다 보니, IE 같은 구형 브라우저에서는 제대로 동작하지 않는다는 문제가 있었습니다. 그러다 보</span>니 <a href="https://github.com/systemjs/systemjs">SystemJS</a>처럼<span style="color:#212529;">CommonJS, AMD, 그리고 최신 ES6까지의 모듈 시스템을 모두 지원하는 또 다른 형태의 모듈 로더가 나오는 웃지 못할 상황도 생기게 되었죠.</span></p><figure class="table" style="text-align:justify;"><table><tbody><tr><td style="background-color:hsl(0, 0%, 90%);"><span style="color:#495057;">여기서 모듈 로더(module loader)라는 용어를 한 번 짚고 넘어가고 싶은데요, 모듈 로더는 <strong>JavaScript 모듈을 런타임에 로드할 수 있게 만드는 구현체</strong>라고 생각하시면 됩니다. 즉, AMD의 모듈 로더는 RequireJS이고 ES6 방식의 모듈 로더는 네이티브 브라우저가 될 수 있겠죠. SystemJS는 모든 모듈을 로드할 수 있는 만큼 ES6, CommonJS, AMD 방식의 모듈 로더입니다.</span></td></tr></tbody></table></figure><p style="text-align:justify;"> </p><figure class="image image_resized" style="width:100%;"><a href="https://www.wishket.com/w/MvLohooMf5"><img src="https://yozm.wishket.com/media/news/1261/yozm-banner-center.png"></a></figure><p style="text-align:justify;"> </p><h3 style="text-align:justify;"><span style="color:#212529;"><strong>트랜스파일러</strong></span></h3><figure class="image image_resized" style="width:100%;"><img src="https://yozm.wishket.com/media/news/1261/image005.png"></figure><p style="text-align:justify;"><span style="color:#212529;">ES6에서 JavaScript 표준 모듈 문법이 정의되었음에도 불구하고, ES6 문법을 구형 브라우저에서 사용하지 못해서 SystemJS 같은 또 다른 라이브러리에 의존해야 하는 것은 꽤나 꺼림칙한 일이었습니다. 그래서 나온 아이디어가 바로 트랜스파일러(Transpiler), 즉 <strong>한 번 컴파일하면 구형 브라우저에서도 동작하는 JavaScript 코드가 나오게 만드는 도구</strong>를 생각하게 되죠.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">가장 유명한 것으로는 바벨(Babel)이 있습니다. 개발할 때에는 최신 JavaScript 문법을 사용하되, 바벨로 컴파일을 하고 난 후에는 같은 동작을 하면서 구형 브라우저 호환이 되는 JavaScript 코드로 변환되니 개발자들은 호환성 걱정 없이 생산성 높은 최신 문법을 사용할 수 있게 되었습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">CoffeeScript, TypeScript 같은 JavaScript의 슈퍼셋(Superset) 언어의 등장 역시 하나의 방법이 되었습니다. 꼭 JavaScript 문법을 쓸 필요가 없기도 하고, 귀찮은 모듈 시스템의 관리는 컴파일러가 대신해 주는 것이죠. 덕분에 호환성에 대한 고민거리는 많이 사라지게 되었습니다.</span></p><hr><p style="text-align:justify;"><span style="color:#212529;">지금까지의 얘기는 JavaScript 모듈화에 대한 이야기였고, 과거로 다시 돌아가서 프론트엔드 생태계 이야기를 해보려고 합니다. 갑자기 이 이야기가 왜 나오냐고요? 바로 웹팩과 같은 모던 모듈 번들러의 등장을 이야기하기 위함입니다. 아까 모듈 시스템이 갈라진 이유가 서버 사이드와 브라우저의 관점이 달랐기 때문이라고 얘기했는데, 지금부터는 좀 더 브라우저, 즉 프론트엔드 관점에서 이야기를 해 볼게요.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"> </p><h3 style="text-align:justify;"><span style="color:#212529;"><strong>태스크 러너</strong></span></h3><figure class="image image_resized" style="width:100%;"><img src="https://yozm.wishket.com/media/news/1261/image006.jpg"></figure><p style="text-align:justify;"><span style="color:#212529;">우리가 지금까지 CommonJS와 AMD, UMD 방식으로 모듈 시스템에 대해 이야기한 것은 바로 스코프가 구분되는 모듈을 만들기 위함이었죠. 그렇다면 스코프가 구분되는 모듈을 만드는 목적은 무엇일까요? 바로 여러 모듈을 조합해 중복되는 코드를 줄이면서도, 생산성과 퍼포먼스가 뛰어난 애플리케이션을 만들기 위함입니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">그러기 위해서는 개발 과정에서 코드를 작성하고, 컨벤션을 유지하기 위해 린트를 사용하며, Sass나 TypeScript처럼 전처리가 필요한 언어를 컴파일하고, 소스 코드를 축소(minify)하고 하나의 파일로 묶는(bundle) 일련의 과정들이 필수적으로 동반됩니다. 이러한 과정들의 반복은 <i>이러한 특정 작업들을 자동화할 수 있는 도구의 필요성</i>으로 이어졌고, 그 결과물이 바로 태스크 러너(task runner)입니다. 즉, 태스크 러너는 <strong>태스크 러너는 프로덕트 개발 과정에서 필요한 일련의 과정들(린팅, 빌딩, 테스팅 등)을 자동화하기 위한 도구</strong>입니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">이러한 태스크 러너들에 대한 필요성이 모듈 시스템의 등장 이후에 등장한 것은 아닙니다. 태스크를 코드로 자동화한다는 생각은 1977년 만들어진 </span><a href="https://en.wikipedia.org/wiki/Make_%28software%29">Make</a><span style="color:#212529;">라는 도구에서도 볼 수 있듯이, 이미 오래 전부터 생각된 것이죠. 하지만 프론트엔드 생태계에서는 크게 필요성을 느끼지 못해서인지, 당시까지만 하더라도 태스크 자동화에 대한 뚜렷한 성과가 있지는 않았습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">그러다가 Node.js와 웹 생태계의 성장으로 인해 프론트엔드에서도 이것저것 처리해야 할 일이 많아졌고, 그래서 한 번쯤은 들어봤</span>을 <a href="https://gruntjs.com/">Grunt</a><span style="color:#212529;">와 </span><a href="https://gulpjs.com/">Gulp</a><span style="color:#212529;">가 등장하게 됩니다. Grunt와 Gulp는 테스트나 린트, 번들, 최적화 플러그인들을 제공해주면서 이 과정들을 자동화할 수 있게 도와주었고 덕분에 많은 프론트엔드 개발자가 사용하게 되죠.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">이 과정에서 빌드를 위한 태스크 러너의 한 과정이었던 번들을 좀 더 전문적으로 도와주는 도구가 필요해졌고, 이것이 바로 모듈 번들러의 등장입니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"> </p><h3 style="text-align:justify;"><span style="color:#212529;"><strong>모듈 번들러</strong></span></h3><figure class="image image_resized" style="width:100%;"><img src="https://yozm.wishket.com/media/news/1261/image007.jpg"><figcaption>번들러는 여행 가방을 싸는 것과 같습니다. 꼭 필요한 짐만 넣어야 효율적이거든요.<br>(이미지 출처: Unsplash)</figcaption></figure><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">모듈 번들러(Module Bundler)는 <strong>JavaScript 모듈을 브라우저에서 실행할 수 있는 단일 JavaScript 파일로 번들링 하는 데 사용되는 프론트엔드 개발 도구</strong>입니다. 모듈 로더와 유사한 부분이 있지만, 모듈 번들러는 코드를 프로덕션 환경에서 사용할 수 있도록 준비하는 것에 더 큰 목적이 있습니다. 또한 모듈 로더는 일반적으로 런타임에 모듈을 가져오기 위한 목적인 반면, 번들러는 빌드 시 모듈을 묶어서 단일 번들 파일로 만들기 때문에 런타임에서 추가적인 로드를 할 필요가 없습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">번들러는 크게 세 가지 이유로 사용하는데, <i>아직까지 모든 브라우저가 모듈 시스템을 완전하게 지원하지 않고</i>, <i>코드의 종속성 관계를 관리하는 데 도움이 되며</i>, <i>종속성 순서, 이미지, CSS 에셋 등을 로드하는 데 도움이 되기 때문</i>입니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">이처럼 번들러가 처음 만들어진 이유는 JavaScript 모듈을 브라우저에서 실행할 수 있는 단일 JavaScript 파일로 번들링 하기 위함이었지만, 단순히 번들하는 것을 넘어서서 사용하지 않는 코드를 제거하는 등의 최적화 작업에 대한 필요성도 높아졌습니다. 이에 구글에서는 2009년 </span><a href="https://developers.google.com/closure/compiler">Closure Compiler</a><span style="color:#212529;">라는 전문적인 JavaScript 최적화 도구를 만들기에 이르죠.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">하지만 최근에는 번들러 자체에서 개발과 빌드, 최적화를 위한 각종 플러그인을 제공하고 있기 때문에 굳이 별도의 태스크 러너나 최적화 도구를 쓰지 않아도 되게 되었습니다. 이것이 바로 모던 웹 애플리케이션을 작성할 때 일반적으로 별도의 추가적인 툴을 쓰지 않는 이유입니다.</span></p><p style="text-align:justify;"><span style="color:#212529;">이러한 변화를 이끌어낸 현대적인 모듈 번들러를 꼽자면, 가장 유명한 3대장이 떠오르죠. 바로 <strong>웹팩(Webpack)</strong>, <strong>롤업(Rollup)</strong>, 그리고 <strong>파셀(Parcel)</strong>입니다. 지금부터는 각각의 특징과 장단점에 대해 간단하게 알아볼게요.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"> </p><h3 style="text-align:justify;"><span style="color:#212529;"><strong>Webpack</strong></span></h3><figure class="image image_resized" style="width:80%;"><img src="https://yozm.wishket.com/media/news/1261/image008.png"></figure><p style="text-align:justify;"><span style="color:#212529;">프론트엔드 개발자라면 한 번쯤은 들어봤을 웹팩은, 오래된 만큼 생태계가 풍부하고 안전성이 뛰어난 번들러입니다. 그래서 서브파티 라이브러리 관리나 CSS 전처리, 이미지 에셋 관리 등에 있어서 다른 번들러보다 강점을 보입니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">코드 스플리팅에 있어서는 롤업과 파셀이 성능적으로는 더 뛰어나지만 아직 안정성 면에서는 웹팩이 더 낫습니다. 일부 리서치에서는 코드 스플리팅이 활성화된 단계에서 빌드 시간이 웹팩이 가장 빠르다고도 하더군요.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">또 다른 웹팩의 특징 중 하나는 웹 애플리케이션에서 사용하는 CSS나 이미지 같은 에셋들을 JavaScript 코드로 변환하고, 이를 분석해서 번들하는 방식을 사용한다는 것입니다. 이 때문인지 웹팩의 구성은 다른 번들러에 비해 설정할 게 많고, 많이 복잡합니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">웹팩에서 제공해주는 개발 서버 역시 다른 번들러에 비해 뛰어납니다. 특히 개발 중 변경사항을 자동으로 새로고침 해주는 라이브 리로딩(Live reloading) 기능과, 새로고침 없이 런타임에 브라우저의 모듈을 업데이트하는 핫 모듈 교체(HMR, Hot Module Replacement) 기능이 그러하죠. 웹팩은 기본적으로 해당 옵션이 활성화된 </span><span style="color:#E83E8C;"><code>webpack-dev-server</code></span><span style="color:#212529;"> 플러그인만 설치하면 되지만, 롤업과 파셀은 별도의 디펜던시와 설정을 추가해주거나 특정 상황에서는 잘 동작하지 않는 경우를 보이기 때문입니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">웹팩에서도 트리 쉐이킹이 지원되기는 하지만, CommonJS 방식으로 모듈을 로드한 부분을 ES6 문법으로 교체해야 하고, </span><span style="color:#E83E8C;"><code>package.json</code></span><span style="color:#212529;"> 파일에 </span><span style="color:#E83E8C;"><code>SideEffects</code></span><span style="color:#212529;"> 플래그를 설정해야 하며, </span><span style="color:#E83E8C;"><code>UglifyJSPlugin</code></span><span style="color:#212529;"><code>, </code></span><span style="color:#E83E8C;"><code>Terser</code></span><span style="color:#212529;"> 같은 별도의 플러그인을 설치하는 것처럼 별도의 추가 작업이 필요해서 조금 번거로움이 있다고는 하네요.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">그리고 ES6 모듈 형태로 빌드 결과물을 출력할 수 없고, 복잡한 문서가 진입장벽을 높인다는 것 역시 단점으로 꼽히고 있습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"> </p><h3 style="text-align:justify;"><span style="color:#212529;"><strong>Rollup</strong></span></h3><figure class="image image_resized" style="width:80%;"><img src="https://yozm.wishket.com/media/news/1261/image009.png"></figure><p style="text-align:justify;"><span style="color:#212529;">롤업은 웹팩과 유사한 모듈 번들러이지만, 가장 큰 차이점을 꼽으라면 ES6 모듈 형식으로 빌드 결과물을 출력할 수 있으므로 이를 라이브러리나 패키지 개발에 활용할 수 있다는 것입니다. 웹팩과 파셀은 자체 로더가 있지만 롤업은 ES6 모듈을 기본으로 따르기 때문입니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">이로 인해 코드 스플리팅 측면에서 다른 번들러와 비교해 강점을 보입니다. 중복 제거에 특화되어 있는데, 특히 진입점(entry point)이 여러 개 있을 때 이 부분이 두드러집니다. 롤업은 진입점이 다르기 때문에 중복해서 번들될 수 있는 부분을 공통되는 알아내고 독립된 모듈로 분리해 낼 수 있습니다. 그래서인지 웹 워커(Web worker)와 메인 쓰레드 간에도 코드 스플리팅이 가능하다고 소개합니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">다만 파일의 해시 캐스캐이딩(hash cascading, 하나의 파일의 해시가 바뀌면 그것을 참조한 파일의 해시도 알아서 바뀜)이 약하다는 점이 약점으로 꼽히고 있습니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"> </p><h3 style="text-align:justify;"><span style="color:#212529;"><strong>Parcel</strong></span></h3><figure class="image image_resized" style="width:80%;"><img src="https://yozm.wishket.com/media/news/1261/image010.png"></figure><p style="text-align:justify;"><span style="color:#212529;">우선 파셀은 별도의 설정 파일 없이도 동작합니다(zero config). 즉, 설치만 하면 별도의 설정 파일 없이 빌드 명령어를 입력해 바로 사용할 수 있습니다. 왜냐하면 파셀은 웹팩과 달리 JavaScript 엔트리 포인트를 지정해주는 것이 아니라, 애플리케이션 진입을 위한 HTML 파일 자체를 읽기 때문입니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">즉 HTML 파일을 순서대로 읽어나가면서 JavaScript, CSS, 이미지 에셋 등을 직접 참조합니다. 웹팩이 에셋을 읽기 위해 JavaScript로 변환하기 위한 이런저런 플러그인과 로더를 깔아야 된다는 것과 비교하자면 파셀은 굉장히 자연스러운 흐름을 타고 있죠.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">사용되지 않는 코드를 제거하는 트리 쉐이킹에서도 강점을 보입니다. 파셀은 ES6 및 CommonJS 모듈 모두에 대해 트리 쉐이킹을 지원합니다. NPM의 라이브러리에 있는 대부분의 코드가 여전히 CommonJS를 사용한다는 것을 감안하면 유용한 기능이죠.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">또한 파셀은 트랜스파일러도 간편하게 설정할 수 있습니다. 일반적으로 번들러는 JavaScript 파일만 읽을 수 있기 때문에, CSS나 이미지 같은 에셋들의 종속성을 인식하고 번들에 추가하려면 트랜스파일러를 사용해야 합니다. 트랜스파일이 필요한 파일 유형을 일일이 설정해야 웹팩 및 롤업과 달리, 파셀은 트랜스파일에 대한 기본 제공을 지원합니다. 즉,</span><span style="color:#E83E8C;"><code>.babelrc</code></span><span style="color:#212529;">, </span><span style="color:#E83E8C;">.<code>postcssrc</code></span><span style="color:#212529;">, </span><span style="color:#E83E8C;">.<code>posthtml</code></span><span style="color:#212529;"> 같은 설정 파일들을 프로젝트 루트 디렉토리에 만들기만 하면, 파셀이 자동으로 파일을 읽어와서 세팅을 다 해주죠.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">게다가 </span><span style="color:#E83E8C;"><code>.babelrc</code></span><span style="color:#212529;"> 로 사용자가 직접 설정한 트랜스파일 외에도, 파셀은 항상 모든 모듈에서 Babel을 사용하여 최신 JavaScript를 브라우저에서 지원하는 형식으로 컴파일합니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">자기네들 말로 코드 스플리팅에도 강점을 보인다고는 하지만, 최근에도 </span><a href="https://github.com/parcel-bundler/parcel/issues/4890">깃헙에서 이슈</a><span style="color:#212529;">가 있는 걸 보면, 그리 안정적이지는 않은 모양입니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">그리고 아직까지 웹팩이나 롤업에 비해 좁은 생태계, 떨어지는 안정성 등이 단점으로 꼽히고 있습니다. 공식 문서가 잘 작성되어 있긴 하지만, 일반적인 케이스만을 다룰 뿐 커스텀한 설정이 필요하다면 설정 파일을 다시 작성해야 하죠.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"> </p><h3 style="text-align:justify;"><span style="color:#212529;"><strong>결론</strong></span></h3><p style="text-align:justify;"><span style="color:#212529;"><i>그래서 번들러는 뭘 써야 하는 건데?</i>라고 생각하실 분이 있을 것 같네요. </span><a href="https://medium.com/better-programming/the-battle-of-bundlers-6333a4e3eda9">제가 찾아본 자료</a><span style="color:#212529;">에 의하면, 기본적으로 프로젝트의 목적에 따라 다릅니다. 일반적으로는 다음의 상황을 기준으로 번들러를 선택하는 것 같습니다.</span></p><p style="text-align:justify;"> </p><ul><li style="text-align:justify;">많은 서드파티를 필요로 하는 복잡한 애플리케이션이라면 <strong>웹팩</strong></li><li style="text-align:justify;">최소한의 서드파티로 라이브러리를 만들고 싶다면 <strong>롤업</strong></li><li style="text-align:justify;">복잡한 설정을 피하고 비교적 간단한 애플리케이션을 만들고 싶다면 <strong>파셀</strong></li></ul><p style="text-align:justify;"> </p><figure class="image image_resized" style="width:100%;"><img src="https://yozm.wishket.com/media/news/1261/image011.png"><figcaption><span style="color:#495057;"><i>web.dev에서 만든 tooling.report</i></span></figcaption></figure><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">이처럼 다양한 빌드 도구의 등장은, 번들러의 선택 과정을 더욱 어렵게 만들었습니다. 그래서 최근 구글에서는 각 빌드 도구 간의 장단점을 비교할 수 있는 </span><a href="https://bundlers.tooling.report/">Tooling.Report</a><span style="color:#212529;">라는 사이트를 만들었다고 하더군요. 각 번들러 별 상세한 장단점이 궁금하신 분들은 한 번 들어가서 구경해보시는 것을 추천합니다.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"> </p><h3 style="text-align:justify;"><span style="color:#212529;"><strong>마무리</strong></span></h3><p style="text-align:justify;"><span style="color:#212529;">지금까지 JavaScript 모듈의 탄생에서부터, 모던 번들러의 비교까지에 대해 알아봤습니다. 글을 잘 읽으셨다면, 다시 처음의 짤에 적힌 설명들이 아마(?) 이해가 가실 거라고 생각합니다. 처음에는 번들러에 대한 비교에서 시작된 글이었는데 적다 보니 깊게 파고 들어가 버렸네요.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;">이렇게 비유를 하다 보면 원래의 의도와는 다르게 해석될 여지가 있어서… 잘못된 정보를 전달할 수 있는 위험이 있을까 봐 많은 리서치를 했습니다. 그럼에도 불구하고 제 배경 지식이 그리 깊지가 않다 보니, 잘못된 정보가 있을 수 있는데 혹시나 발견하신다면 알려주세요.</span></p><p style="text-align:justify;"> </p><p style="text-align:justify;"> </p><p style="text-align:justify;"><span style="color:#212529;"><strong>참고자료</strong></span></p><ul><li style="text-align:justify;"><a href="https://www.notion.so/ef972717404442e9ad4a2a804506cc1a#6e057e2cd35045e9b8a80ffe3c7d887d">web.dev LIVE</a></li><li style="text-align:justify;"><a href="https://medium.com/@ajmeyghani/javascript-bundlers-a-comparison-e63f01f2a364">JavaScript Bundlers, a Comparison</a></li><li style="text-align:justify;"><a href="https://survivejs.com/webpack/appendices/comparison/">Comparison of Build Tools</a></li><li style="text-align:justify;"><a href="https://www.infoq.com/news/2020/07/google-team-build-tool-benchmark/">Webpack vs. Rollup vs. Parcel vs. Browserify: a Detailed Benchmark</a></li><li style="text-align:justify;"><a href="https://medium.com/better-programming/the-battle-of-bundlers-6333a4e3eda9">Rollup vs. Parcel vs. webpack: Which Is the Best Bundler?</a></li><li style="text-align:justify;"><a href="https://www.freecodecamp.org/news/anatomy-of-js-module-systems-and-building-libraries-fadcd8dbd0e/">Learn the basics of the JavaScript module system and build your own library</a></li><li style="text-align:justify;"><a href="https://lihautan.com/what-is-module-bundler-and-how-does-it-work/">What is module bundler and how does it work?</a></li><li style="text-align:justify;"><a href="https://dev.to/iggredible/what-the-heck-are-cjs-amd-umd-and-esm-ikm">What are CJS, AMD, UMD, and ESM in Javascript?</a></li><li style="text-align:justify;"><a href="https://stackoverflow.com/questions/33561272/task-runners-gulp-grunt-etc-and-bundlers-webpack-browserify-why-use-toge">Task Runners (Gulp, Grunt, etc) and Bundlers (Webpack, Browserify). Why use together?</a></li></ul>