Will Schenk

howto

How to use pako with vite

javascript string encoding is simply insane

pako is a javascript implementation of the zlib library. I was trying to make it work when I was recreating the notepadtab but I was unable to make it work. Turns out that it's because I have no idea how JavaScript strings work. Not sure that it's actually clearer now, but this code works.

To use pako, pass in the source data to pako.deflate. This will give you a Uint8Array, which you need to convert back to a string for window.btoa to work – otherwise it will encode the string representation of the array. Use String.fromCharCode to make that work.

To get things back, you need to do atob to convert the base64 back to an array of bytes. We then convert that to a string, and then that into a Uint8 array, and inflate that using pako and then turn that again into a string!

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
  import pako from 'pako';

  window.addEventListener("load", (event) => {
      const source = document.getElementById( "source" );
      const encoding = document.getElementById( "encoding" );
      const dest = document.getElementById( "dest" );
      
      source.addEventListener( "keyup", (e) => {
          // Compress and encode
          const inputData = source.value;
          console.log( "inputData", inputData );

          const compressed = pako.deflate( inputData );
          console.log( "compressed", compressed );

          const base64 = window.btoa( String.fromCharCode.apply(null, compressed ));
          encoding.innerHTML = base64;
          console.log( "base64", base64 );

          // Stats
          rawsize.innerHTML = inputData.length;
          hashsize.innerHTML = base64.length;

          // Decompress and inflate back to the original string
          const reverseBase64 = atob(base64);

          // Converts the string back to an Uint8Array
          const reverseBase64Array = new Uint8Array(reverseBase64.split("").map(function(c) {
              return c.charCodeAt(0); }));
          console.log( "reverse", reverseBase64Array );

          const inflatedRaw = pako.inflate( reverseBase64Array );
          console.log( "inflatedRaw", inflatedRaw );


          // Convert back to string
          const decompressed = String.fromCharCode.apply( null, inflatedRaw );
          console.log( "decompressed", decompressed );
          dest.innerHTML = decompressed;
      } )
  } )

Boiler plate

Then create a package.json like this:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
  {
      "type": "module",
      "scripts": {
          "dev": "vite"
      },
      "dependencies": {
          "pako": "^2.1.0",
          "vite": "^5.3.0"
      }
  }
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
  <html>
    <head>
      <title>Pako Test</title>
      <script src="main.js" type="module"></script>
      <meta name="viewport" content="width=device-width, initial-scale=1" />
      <style> 
        textarea {
          width: 100%;
          height: 150px;
          padding: 1em;
        }

        div, pre {
          padding-bottom: 1em;
        }
      </style>
    </head>
    <body >

      <h1>Enter text</h1>
      <div>
        <textarea id="source">This is the source text, edit it</textarea>
      </div>

      <h2>Encoded</h2>
      <pre id="encoding"></pre>

      <div>
        <h2>Size</h2>

        <p>Raw: <span id="rawsize"></span></p>
        <p>Compressed base 64'd: <span id="hashsize"></span></p>
      </div>

      <h2>Back to source</h2>
      <pre id="dest"></pre>
    </body>
  </html>

Test

1
  npm i && npm run dev

And open the console.

Previously

labnotes

Checking fetch for error why is this so hard

Next

fragments

Four freedoms