An eternal token of love, OLGA’s image is stored on the most permanent medium known to man – the Bitcoin blockchain.
Here I will explain how the image is encoded, present a script which retrieves the image from the blockchain, and how to upscale the image.
The Blockchain Image
This exact image is saved on the blockchain:
The file is a miniature png file. A very small size was required to have it fit inside a bitcoin transaction.
Upscaled Version
Apply color grading and cubic resizing and – voila – the image looks more like the original drawing:
The algorithm is called Mantiuk Tone Mapping. I’ll explain the process at the end of this article. But first let’s retrieve the original file.
Base64 Encoding
The image is stored on the blockchain as a base64 string with the recognizable data:image/png;base64
prefix.
I predicted that if someone ever were to discover this string, they would easily be able to regenerate the image. Indeed, this is exactly what happened when OLGA was ‘unearthed’ in 2021.
Bitcoin Transaction
The base64 string is included in a transaction with ID 627ae48d6b4cffb2ea734be1016dedef4cee3f8ffefaea5602dd58c696de6b74
.
You can look it up in any Bitcoin explorer: BlockCypher – Blockchain.com – Blockstream
The transaction has multiple outputs. Here’s the first of them:
The hex codes in blue and yellow contain some of the data. However, it took a full 165 such outputs to to fit the entire image. All of these are concatenated into one long hex string.
RC4 Stream Cipher
The string is RC4 encoded. It must be decoded with the input coin as the key. The input can be seen on Blockstream’s explorer:
Hex To String
The last step is to convert the hex string into a text string. Now you’ll find that it begins with CNTRPRTY
and several more characters before data:image/png;base64,
followed by the image data.
CNTRPRTY is the prefix which indicates the data is Counterparty encoded. This is a standard for encoding token data on Bitcoin. For this article this is irrelevant since I only want to show that OLGA’s image can be retrieved directly from Bitcoin. However, it’s worth mentioning that anyone running a Bitcoin node with the Counterparty extension can open the database and view OLGA’s code there.
String To Image
As mentioned before, this string can be opened with HTML. Include it in an img
tag.
Save the below code as olga.html
and open it in a browser:
<!DOCTYPE html> <html> <head><title>OLGA</title></head> <body> <div style="text-align: center;"> <h1>OLGA</h1> <img width="178" height="130" src=""> </div> </body> </html>
Raw Transaction To Image
The below code generates OLGA’s image from the raw Bitcoin transaction. It gathers the data from Blockcypher’s API, decodes, converts to text, extracts the image data and finally displays the image.
Note that no libraries are needed. Only some RC4 helper functions are required. The code will stop working if Blockcypher goes down, but the same data can be collected from any Bitcoin node. All of them keep an exact copy. That’s the entire point of Bitcoin – and Bitcoin NFTs!
<!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=utf-8"> <title>OLGA</title> <script> let tx = '627ae48d6b4cffb2ea734be1016dedef4cee3f8ffefaea5602dd58c696de6b74' async function get(tx) { let url = 'https://api.blockcypher.com/v1/btc/main/txs/' + tx + '?instart=0&outstart=0&limit=200'; let obj = await (await fetch(url)).json(); return obj; } let json; setTimeout(() => { get_json(tx); }, 1402); function get_json(tx) { (async () => { json = await get(tx) decode(json); })() } function decode(json) { //GET INPUT ID AND OUTPUT SCRIPTS json = JSON.parse(JSON.stringify(json)); let utxo = json['inputs'][0]['prev_hash']; let script = []; for (const x of json['outputs']) { if (x['script_type'] == 'pay-to-multi-pubkey-hash') { script.push(x['script']); } } //HEX DATA FROM BITCOIN OUTPUTS let hex_data = ''; for (let i = 0; i < script.length; i++) { let raw = script[i].substring(6, 68) + script[i].substring(74, 136); raw = xcp_rc4(utxo, raw); let len = raw.substring(0, 2); len = parseInt(len, 16); raw = raw.substring(2, 2+(len*2)); raw = raw.substring(16); hex_data += raw; } //HEX TO ASCII STRING let raw_ascii = hex2a(hex_data); //IMAGE DATA FROM ASCII STRING let img_data = raw_ascii.substring(raw_ascii.indexOf('data:image/')); img_data = img_data.slice(0,-1); //OUTPUT HTML let img_tag = '<img src="' + img_data + '">'; let img_tag_double = '<img width="178" height="130" src="' + img_data + '">'; let html = '<h1>OLGA</h1>'; html += '<p><br><b>ORIGINAL</b><br>(89x65)<br>' + img_tag + '</p>'; html += '<p><br><b>DOUBLE</b><br>(178x130)<br>' + img_tag_double + '</p>'; document.getElementById('content').innerHTML = html; } //HELPER FUNCTIONS function hex2a(hexx) { var hex = hexx.toString();//force conversion var str = ''; for (var i = 0; i < hex.length; i += 2) str += String.fromCharCode(parseInt(hex.substr(i, 2), 16)); return str; } function xcp_rc4(key, datachunk) { return bin2hex(rc4(hex2bin(key), hex2bin(datachunk))); } function hex2bin(hex) { var bytes = []; var str; for (var i = 0; i < hex.length - 1; i += 2) { var ch = parseInt(hex.substr(i, 2), 16); bytes.push(ch); } str = String.fromCharCode.apply(String, bytes); return str; }; function bin2hex(s) { // http://kevin.vanzonneveld.net var i, l, o = "", n; s += ""; for (i = 0, l = s.length; i < l; i++) { n = s.charCodeAt(i).toString(16); o += n.length < 2 ? "0" + n : n; } return o; }; function rc4(key, str) { //https://gist.github.com/farhadi/2185197 var s = [], j = 0, x, res = ''; for (var i = 0; i < 256; i++) { s[i] = i; } for (i = 0; i < 256; i++) { j = (j + s[i] + key.charCodeAt(i % key.length)) % 256; x = s[i]; s[i] = s[j]; s[j] = x; } i = 0; j = 0; for (var y = 0; y < str.length; y++) { i = (i + 1) % 256; j = (j + s[i]) % 256; x = s[i]; s[i] = s[j]; s[j] = x; res += String.fromCharCode(str.charCodeAt(y) ^ s[(s[i] + s[j]) % 256]); } return res; } </script> </head> <body> <div id="content" style="text-align: center;"> <h2>Loading Blockcypher API..</h2> </div> </body> </html>
Upscaling with GIMP
The blokchain image was overexposed on purpose. The intent was to make a near-invisible engraving that would pop out when looking from an angle. On a modern screen this may not hold true anymore, but that’s how I presented it to Olga on our old LCD monitor.
Anyway, for showcasing OLGA it may be desired to apply a filter and increase the resolution.
Thankfully, this is easy with GIMP;
- Image → Crop to Content
The blockchain image had the last few characters cut off due to a wallet bug. Therefore the bottom line is shown as transparent pixels. This command simply removes that line. - Image → Tone Mapping → Mantiuk 2006…
Leave the default settings and click OK. - Image → Scale Image
Set Width and Height to 300% and Interpolation to Cubic.
That’s all!
It’s on my TODO to do this programmatically with the gimpfu library.
Categories: Uncategorized