<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Oleksandr</title>
    <description>The latest articles on DEV Community by Oleksandr (@olek3).</description>
    <link>https://dev.to/olek3</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3984904%2Ffc9eea2a-8c30-4c46-b2d4-5f43946cfad6.png</url>
      <title>DEV Community: Oleksandr</title>
      <link>https://dev.to/olek3</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/olek3"/>
    <language>en</language>
    <item>
      <title>Reviving a Dead Plugin: How I Bring Back To Life the serverless-s3-sync plugin</title>
      <dc:creator>Oleksandr</dc:creator>
      <pubDate>Mon, 15 Jun 2026 10:05:04 +0000</pubDate>
      <link>https://dev.to/olek3/reviving-a-dead-plugin-how-i-bring-back-to-life-the-serverless-s3-sync-plugin-1lgn</link>
      <guid>https://dev.to/olek3/reviving-a-dead-plugin-how-i-bring-back-to-life-the-serverless-s3-sync-plugin-1lgn</guid>
      <description>&lt;p&gt;If you've been using the Serverless Framework to deploy static assets to S3, you've probably used &lt;a href="https://www.npmjs.com/package/serverless-s3-sync" rel="noopener noreferrer"&gt;serverless-s3-sync&lt;/a&gt;. It was the go-to plugin for syncing local directories to S3 buckets. Simple config, works after sls deploy, done.&lt;/p&gt;

&lt;p&gt;Then, on January 1, 2026, the original author archived the repository. No more updates, no more security fixes. The most worrying thing - no more security fixes.&lt;/p&gt;

&lt;p&gt;So I forked it, cleaned it up, and published it as &lt;a href="https://www.npmjs.com/package/serverless-s3-sync-v2" rel="noopener noreferrer"&gt;serverless-s3-sync-v2&lt;/a&gt;.&lt;/p&gt;

&lt;h2&gt;
  
  
  Original Plugin Had a Problem
&lt;/h2&gt;

&lt;p&gt;The original &lt;a href="https://www.npmjs.com/package/serverless-s3-sync" rel="noopener noreferrer"&gt;serverless-s3-sync&lt;/a&gt; worked, but its dependency tree was a historical artifact:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight json"&gt;&lt;code&gt;&lt;span class="err"&gt;//&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="err"&gt;Original&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="err"&gt;dependencies&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="err"&gt;(k&lt;/span&gt;&lt;span class="mi"&gt;1&lt;/span&gt;&lt;span class="err"&gt;LoW/serverless-s&lt;/span&gt;&lt;span class="mi"&gt;3&lt;/span&gt;&lt;span class="err"&gt;-sync)&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"@auth0/s3"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"^1.0.0"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"bluebird"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"^3.5.1"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"mime"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"^2.4.0"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"minimatch"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"^3.0.4"&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;a href="https://www.npmjs.com/package/@auth0/s3" rel="noopener noreferrer"&gt;@auth0/s3&lt;/a&gt; is itself a fork of the andrewrk/node-s3-client library, which hasn't been maintained for years. It's built on top of AWS SDK v2, which is also deprecated and has end-of-support on September 8, 2025. It handled the entire upload logic: directory scanning, MD5 diffing, multipart uploads, and the actual putObject calls.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://www.npmjs.com/package/bluebird" rel="noopener noreferrer"&gt;bluebird&lt;/a&gt; was a Promise polyfill that made sense in 2015, when native Promises were slow or unreliable in Node.js. In 2026, it's pure dead weight.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;The result&lt;/strong&gt;: to sync a folder to S3, you were pulling in an unmaintained S3 client library that wrapped an unmaintained Promise library that wrapped an outdated AWS SDK. Four layers of abandoned code between you and a simple PutObject API call.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Fork: What Changed
&lt;/h2&gt;

&lt;ol&gt;
&lt;li&gt;Replaced @auth0/s3 + bluebird with direct AWS SDK v3&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;The most significant change. Instead of delegating to a middleware library, the fork talks directly to @aws-sdk/client-s3:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// Before (via @auth0/s3 abstraction)&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;client&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;s3&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;createClient&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt; &lt;span class="na"&gt;s3Options&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;awsOptions&lt;/span&gt; &lt;span class="p"&gt;});&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;uploader&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;client&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;uploadDir&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt; &lt;span class="nx"&gt;localDir&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;s3Params&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;Bucket&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;Prefix&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="p"&gt;});&lt;/span&gt;

&lt;span class="c1"&gt;// After (direct AWS SDK v3)&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;S3Client&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;PutObjectCommand&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;ListObjectsV2Command&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;DeleteObjectsCommand&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;@aws-sdk/client-s3&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;client&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;S3Client&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;awsOptions&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;client&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;send&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;PutObjectCommand&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt; &lt;span class="nx"&gt;Bucket&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;Key&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;Body&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;ContentType&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="p"&gt;...&lt;/span&gt;&lt;span class="nx"&gt;params&lt;/span&gt; &lt;span class="p"&gt;}));&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;AWS SDK v3 is modular — you import only what you need. No more pulling in the entire SDK.&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Rewrote sync logic with native async/await&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;&lt;a href="https://www.npmjs.com/package/bluebird" rel="noopener noreferrer"&gt;bluebird&lt;/a&gt; is gone. All async operations use native Promise and async/await. The Node.js built-in crypto module handles MD5 hashing for ETag comparison:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;crypto&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;crypto&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;

&lt;span class="kd"&gt;function&lt;/span&gt; &lt;span class="nf"&gt;computeMD5&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;filePath&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;Promise&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;resolve&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;reject&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;hash&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;crypto&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;createHash&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;md5&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
    &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;stream&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;fs&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;createReadStream&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;filePath&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
    &lt;span class="nx"&gt;stream&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;on&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;data&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;chunk&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nx"&gt;hash&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;update&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;chunk&lt;/span&gt;&lt;span class="p"&gt;));&lt;/span&gt;
    &lt;span class="nx"&gt;stream&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;on&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;end&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="p"&gt;()&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nf"&gt;resolve&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;hash&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;digest&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;hex&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;)));&lt;/span&gt;
    &lt;span class="nx"&gt;stream&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;on&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;error&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;reject&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
  &lt;span class="p"&gt;});&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;ol&gt;
&lt;li&gt;Node.js ≥ 20 requirement&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;The fork sets &lt;code&gt;"node": "&amp;gt;=20"&lt;/code&gt; in engines. This enables native test runner (&lt;code&gt;node --test&lt;/code&gt;), native fetch, and all modern async primitives without polyfills.&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Updated mime and minimatch&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;&lt;a href="https://www.npmjs.com/package/mime" rel="noopener noreferrer"&gt;mime&lt;/a&gt; upgraded from v2 to v4 (ESM-aware, smaller, maintained)&lt;br&gt;
&lt;a href="https://www.npmjs.com/package/minimatch" rel="noopener noreferrer"&gt;minimatch&lt;/a&gt; upgraded from v3 to v10&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Added ESLint with standard config&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;The original repo lacks adding a consistent code style enforced via a linter. For linting JavaScript code, I added &lt;a href="https://www.npmjs.com/package/standard" rel="noopener noreferrer"&gt;standard&lt;/a&gt;. For now, the standard hasn't been updated for 2 years, and I hope it will be brought back to life soon.&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Native Node.js test runner&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;Tests migrated from no tests to &lt;code&gt;node --test&lt;/code&gt;, removing the need for a separate test framework dependency. A lot of unit tests were added to confirm the stability of the code.&lt;/p&gt;

&lt;h2&gt;
  
  
  Migration from the Original
&lt;/h2&gt;

&lt;p&gt;If you're already using &lt;a href="https://www.npmjs.com/package/serverless-s3-sync" rel="noopener noreferrer"&gt;serverless-s3-sync&lt;/a&gt;, migration is two steps:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Swap the package:
&lt;/li&gt;
&lt;/ol&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npm uninstall serverless-s3-sync
npm &lt;span class="nb"&gt;install&lt;/span&gt; &lt;span class="nt"&gt;--save&lt;/span&gt; serverless-s3-sync-v2
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;ol&gt;
&lt;li&gt;Update serverless.yml:
&lt;/li&gt;
&lt;/ol&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="na"&gt;plugins&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;serverless-s3-sync-v2&lt;/span&gt;  &lt;span class="c1"&gt;# was: serverless-s3-sync&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Your existing custom.s3Sync configuration stays exactly the same. All options are preserved:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;code&gt;bucketName&lt;/code&gt; / &lt;code&gt;bucketNameKey&lt;/code&gt;
&lt;/li&gt;
&lt;li&gt;&lt;code&gt;bucketPrefix&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;&lt;code&gt;localDir&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;&lt;code&gt;deleteRemoved&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;&lt;code&gt;acl&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;params&lt;/code&gt; (per-file headers like &lt;code&gt;CacheControl&lt;/code&gt;)&lt;/li&gt;
&lt;li&gt;&lt;code&gt;bucketTags&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;enabled&lt;/code&gt; (conditional sync rules)&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;--nos3sync&lt;/code&gt; CLI flag&lt;/li&gt;
&lt;li&gt;Custom &lt;code&gt;hooks&lt;/code&gt;
&lt;/li&gt;
&lt;li&gt;Offline mode via &lt;code&gt;serverless-s3-local&lt;/code&gt;
&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Links
&lt;/h2&gt;

&lt;p&gt;npm: &lt;a href="https://www.npmjs.com/package/serverless-s3-sync-v2" rel="noopener noreferrer"&gt;serverless-s3-sync-v2&lt;/a&gt;&lt;br&gt;
GitHub: &lt;a href="https://github.com/AlexHladin/serverless-s3-sync" rel="noopener noreferrer"&gt;AlexHladin/serverless-s3-sync&lt;/a&gt;&lt;br&gt;
Original (archived): &lt;a href="https://github.com/k1LoW/serverless-s3-sync" rel="noopener noreferrer"&gt;k1LoW/serverless-s3-sync&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;If this saved you from a supply chain dependency nightmare, give it a ⭐ on GitHub. PRs and issues welcome.&lt;/p&gt;

</description>
      <category>serverless</category>
      <category>aws</category>
      <category>webdev</category>
      <category>opensource</category>
    </item>
  </channel>
</rss>
