Avatar
b
87ae535006d142ab4ffb28c964241bca3a2f98801f5ce6282cd97f8b8d6b32ed

flare.pub is probably ok

its pretty much same as lbry

lbry never solved the issue of content hosting

with nostr we have like 6 centralized http servers so far

but it will get better

consider this

events should have expiration by default

there is no use to store all spam forever

expiration could be disabled by providing certain pow

#nostr #dev

Replying to Avatar b

#publishevent

data:text/html;base64,<!DOCTYPE html>
<html lang="">
  <head>
    <meta charset="utf-8">
    <title>publishevent</title>
    <style>
    textarea {
      font-size: 1.2em;
      width: 50%;
      height: 30em;
      box-sizing: border-box;
      tab-size : 2;
      resize: vertical;
    }
    #response {
      float: right;
      width: calc( 50% - 2em );
      padding: 0 1em;
      word-break: break-word;
    }

    #response div,
    #response a {
      margin-bottom: 1em;
    }
    </style>
    <script src="https://unpkg.com/nostr-tools/lib/nostr.bundle.js" crossorigin="anonymous"></script>
    <script>
    (function (exports) {
      'use strict';

      function insertTextIntoField(field, text) {
          const document = field.ownerDocument;
          const initialFocus = document.activeElement;
          if (initialFocus !== field) {
              field.focus();
          }
          if (text === '') {
              document.execCommand('delete');
          }
          else {
              document.execCommand('insertText', false, text);
          }
          if (initialFocus === document.body) {
              field.blur();
          }
          else if (initialFocus instanceof HTMLElement && initialFocus !== field) {
              initialFocus.focus();
          }
      }

      function indentSelection(element) {
          const { selectionStart, selectionEnd, value } = element;
          const selectedText = value.slice(selectionStart, selectionEnd);
          const lineBreakCount = /\n/g.exec(selectedText)?.length;
          if (lineBreakCount > 0) {
              const firstLineStart = value.lastIndexOf('\n', selectionStart - 1) + 1;
              const newSelection = element.value.slice(firstLineStart, selectionEnd - 1);
              const indentedText = newSelection.replaceAll(/^|\n/g,
              '$&\t');
              const replacementsCount = indentedText.length - newSelection.length;
              element.setSelectionRange(firstLineStart, selectionEnd - 1);
              insertTextIntoField(element, indentedText);
              element.setSelectionRange(selectionStart + 1, selectionEnd + replacementsCount);
          }
          else {
              insertTextIntoField(element, '\t');
          }
      }
      function findLineEnd(value, currentEnd) {
          const lastLineStart = value.lastIndexOf('\n', currentEnd - 1) + 1;
          if (value.charAt(lastLineStart) !== '\t') {
              return currentEnd;
          }
          return lastLineStart + 1;
      }
      function unindentSelection(element) {
          const { selectionStart, selectionEnd, value } = element;
          const firstLineStart = value.lastIndexOf('\n', selectionStart - 1) + 1;
          const minimumSelectionEnd = findLineEnd(value, selectionEnd);
          const newSelection = element.value.slice(firstLineStart, minimumSelectionEnd);
          const indentedText = newSelection.replaceAll(/(^|\n)(\t| {1,2})/g, '$1');
          const replacementsCount = newSelection.length - indentedText.length;
          element.setSelectionRange(firstLineStart, minimumSelectionEnd);
          insertTextIntoField(element, indentedText);
          const firstLineIndentation = /\t| {1,2}/.exec(value.slice(firstLineStart, selectionStart));
          const difference = firstLineIndentation
              ? firstLineIndentation[0].length
              : 0;
          const newSelectionStart = selectionStart - difference;
          element.setSelectionRange(selectionStart - difference, Math.max(newSelectionStart, selectionEnd - replacementsCount));
      }
      function tabToIndentListener(event) {
          if (event.defaultPrevented
              || event.metaKey
              || event.altKey
              || event.ctrlKey) {
              return;
          }
          const textarea = event.target;
          if (event.key === 'Tab') {
              if (event.shiftKey) {
                  unindentSelection(textarea);
              }
              else {
                  indentSelection(textarea);
              }
              event.preventDefault();
              event.stopImmediatePropagation();
          }
          else if (event.key === 'Escape'
              && !event.shiftKey) {
              textarea.blur();
              event.preventDefault();
              event.stopImmediatePropagation();
          }
      }
      function enableTabToIndent(elements, signal) {
          if (typeof elements === 'string') {
              elements = document.querySelectorAll(elements);
          }
          else if (elements instanceof HTMLTextAreaElement) {
              elements = [elements];
          }
          for (const element of elements) {
              element.addEventListener('keydown', tabToIndentListener, { signal });
          }
      }
      const indent = indentSelection;
      const unindent = unindentSelection;
      const eventHandler = tabToIndentListener;
      const watch = enableTabToIndent;

      exports.enableTabToIndent = enableTabToIndent;
      exports.eventHandler = eventHandler;
      exports.indent = indent;
      exports.indentSelection = indentSelection;
      exports.tabToIndentListener = tabToIndentListener;
      exports.unindent = unindent;
      exports.unindentSelection = unindentSelection;
      exports.watch = watch;

      Object.defineProperty(exports, '__esModule', { value: true });
    }(this.window = this.window || {}));
    </script>
  </head>
  <body>
    <form action="#">
      <div id="response">
        <div id="response-raw"></div>
        <a id="response-link" href="#"></a>
      </div>
      <p>
        <textarea></textarea>
      </p>
      <p>
        apply template:
        <a href="#" data-template="1">kind 1</a>,
        <a href="#" data-template="1063">kind 1063</a>
      </p>
      <p>
        <input type="submit" value="publish"/>
      </p>
    </form>
    <script>
      const response_link = document.querySelector("#response-link")
      const textarea = document.querySelector("textarea")
      const templates = {}

      templates[1] = {
        "kind": 1,
        "content": "",
        "tags": []
      }

      templates[1063] = {
        "kind": 1063,
        "tags": [
          ["url",""]
        ],
        "content": "",
      }

      for(let key in templates){
        templates[key] = JSON.stringify(templates[key], null, "\t")
      }



      function apply_template(link){
        textarea.value = templates[link.dataset.template]
      }

      for(let template_link of document.querySelectorAll("a[data-template]")){
        template_link.onclick = function(e){
          e.preventDefault()
          apply_template(this)
        }
      }

      apply_template(document.querySelector("a[data-template]"))
      enableTabToIndent(textarea)

      document.querySelector("form").onsubmit = function(e){
        e.preventDefault()

        try{
          const event = JSON.parse(textarea.value)
          event.created_at = event.created_at || Math.floor(new Date().getTime() / 1000)

          const sk = NostrTools.generateSecretKey()
          NostrTools.finalizeEvent(event, sk)

          const socket = new WebSocket("wss://nos.lol")

          socket.onopen = function(){
            socket.send(JSON.stringify(["EVENT", event]))
          }

          socket.onmessage = function(e){
            document.querySelector("#response-raw").innerText = e.data

            const data = JSON.parse(e.data)

            if(data[0] === "OK"){
              console.log("id", data[1])
              //const note_id = NostrTools.nip19.npubEncode(data[1])
              const nevent = NostrTools.nip19.neventEncode(event)

              response_link.href = response_link.innerText = "https://coracle.social/" + nevent
            }

            socket.close()
          }
        }catch(e){
          document.querySelector("#response").innerText = e.message
        }
      }
    </script>
  </body>
</html>


no one invented this before

simplest possible way to publish json events

even simplest possible way to publish kind 1 note

just fill in content, press send, its published

change event kind, change other properties etc.

nostr:note1fk53prrks6zdt0ghakdqdtsuhejryda8ha854hr3hulkuvdl8juqgfz0nd

#publishevent

data:text/html;base64,<!DOCTYPE html>
<html lang="">
  <head>
    <meta charset="utf-8">
    <title>publishevent</title>
    <style>
    textarea {
      font-size: 1.2em;
      width: 50%;
      height: 30em;
      box-sizing: border-box;
      tab-size : 2;
      resize: vertical;
    }
    #response {
      float: right;
      width: calc( 50% - 2em );
      padding: 0 1em;
      word-break: break-word;
    }

    #response div,
    #response a {
      margin-bottom: 1em;
    }
    </style>
    <script src="https://unpkg.com/nostr-tools/lib/nostr.bundle.js" crossorigin="anonymous"></script>
    <script>
    (function (exports) {
      'use strict';

      function insertTextIntoField(field, text) {
          const document = field.ownerDocument;
          const initialFocus = document.activeElement;
          if (initialFocus !== field) {
              field.focus();
          }
          if (text === '') {
              document.execCommand('delete');
          }
          else {
              document.execCommand('insertText', false, text);
          }
          if (initialFocus === document.body) {
              field.blur();
          }
          else if (initialFocus instanceof HTMLElement && initialFocus !== field) {
              initialFocus.focus();
          }
      }

      function indentSelection(element) {
          const { selectionStart, selectionEnd, value } = element;
          const selectedText = value.slice(selectionStart, selectionEnd);
          const lineBreakCount = /\n/g.exec(selectedText)?.length;
          if (lineBreakCount > 0) {
              const firstLineStart = value.lastIndexOf('\n', selectionStart - 1) + 1;
              const newSelection = element.value.slice(firstLineStart, selectionEnd - 1);
              const indentedText = newSelection.replaceAll(/^|\n/g,
              '$&\t');
              const replacementsCount = indentedText.length - newSelection.length;
              element.setSelectionRange(firstLineStart, selectionEnd - 1);
              insertTextIntoField(element, indentedText);
              element.setSelectionRange(selectionStart + 1, selectionEnd + replacementsCount);
          }
          else {
              insertTextIntoField(element, '\t');
          }
      }
      function findLineEnd(value, currentEnd) {
          const lastLineStart = value.lastIndexOf('\n', currentEnd - 1) + 1;
          if (value.charAt(lastLineStart) !== '\t') {
              return currentEnd;
          }
          return lastLineStart + 1;
      }
      function unindentSelection(element) {
          const { selectionStart, selectionEnd, value } = element;
          const firstLineStart = value.lastIndexOf('\n', selectionStart - 1) + 1;
          const minimumSelectionEnd = findLineEnd(value, selectionEnd);
          const newSelection = element.value.slice(firstLineStart, minimumSelectionEnd);
          const indentedText = newSelection.replaceAll(/(^|\n)(\t| {1,2})/g, '$1');
          const replacementsCount = newSelection.length - indentedText.length;
          element.setSelectionRange(firstLineStart, minimumSelectionEnd);
          insertTextIntoField(element, indentedText);
          const firstLineIndentation = /\t| {1,2}/.exec(value.slice(firstLineStart, selectionStart));
          const difference = firstLineIndentation
              ? firstLineIndentation[0].length
              : 0;
          const newSelectionStart = selectionStart - difference;
          element.setSelectionRange(selectionStart - difference, Math.max(newSelectionStart, selectionEnd - replacementsCount));
      }
      function tabToIndentListener(event) {
          if (event.defaultPrevented
              || event.metaKey
              || event.altKey
              || event.ctrlKey) {
              return;
          }
          const textarea = event.target;
          if (event.key === 'Tab') {
              if (event.shiftKey) {
                  unindentSelection(textarea);
              }
              else {
                  indentSelection(textarea);
              }
              event.preventDefault();
              event.stopImmediatePropagation();
          }
          else if (event.key === 'Escape'
              && !event.shiftKey) {
              textarea.blur();
              event.preventDefault();
              event.stopImmediatePropagation();
          }
      }
      function enableTabToIndent(elements, signal) {
          if (typeof elements === 'string') {
              elements = document.querySelectorAll(elements);
          }
          else if (elements instanceof HTMLTextAreaElement) {
              elements = [elements];
          }
          for (const element of elements) {
              element.addEventListener('keydown', tabToIndentListener, { signal });
          }
      }
      const indent = indentSelection;
      const unindent = unindentSelection;
      const eventHandler = tabToIndentListener;
      const watch = enableTabToIndent;

      exports.enableTabToIndent = enableTabToIndent;
      exports.eventHandler = eventHandler;
      exports.indent = indent;
      exports.indentSelection = indentSelection;
      exports.tabToIndentListener = tabToIndentListener;
      exports.unindent = unindent;
      exports.unindentSelection = unindentSelection;
      exports.watch = watch;

      Object.defineProperty(exports, '__esModule', { value: true });
    }(this.window = this.window || {}));
    </script>
  </head>
  <body>
    <form action="#">
      <div id="response">
        <div id="response-raw"></div>
        <a id="response-link" href="#"></a>
      </div>
      <p>
        <textarea></textarea>
      </p>
      <p>
        apply template:
        <a href="#" data-template="1">kind 1</a>,
        <a href="#" data-template="1063">kind 1063</a>
      </p>
      <p>
        <input type="submit" value="publish"/>
      </p>
    </form>
    <script>
      const response_link = document.querySelector("#response-link")
      const textarea = document.querySelector("textarea")
      const templates = {}

      templates[1] = {
        "kind": 1,
        "content": "",
        "tags": []
      }

      templates[1063] = {
        "kind": 1063,
        "tags": [
          ["url",""]
        ],
        "content": "",
      }

      for(let key in templates){
        templates[key] = JSON.stringify(templates[key], null, "\t")
      }



      function apply_template(link){
        textarea.value = templates[link.dataset.template]
      }

      for(let template_link of document.querySelectorAll("a[data-template]")){
        template_link.onclick = function(e){
          e.preventDefault()
          apply_template(this)
        }
      }

      apply_template(document.querySelector("a[data-template]"))
      enableTabToIndent(textarea)

      document.querySelector("form").onsubmit = function(e){
        e.preventDefault()

        try{
          const event = JSON.parse(textarea.value)
          event.created_at = event.created_at || Math.floor(new Date().getTime() / 1000)

          const sk = NostrTools.generateSecretKey()
          NostrTools.finalizeEvent(event, sk)

          const socket = new WebSocket("wss://nos.lol")

          socket.onopen = function(){
            socket.send(JSON.stringify(["EVENT", event]))
          }

          socket.onmessage = function(e){
            document.querySelector("#response-raw").innerText = e.data

            const data = JSON.parse(e.data)

            if(data[0] === "OK"){
              console.log("id", data[1])
              //const note_id = NostrTools.nip19.npubEncode(data[1])
              const nevent = NostrTools.nip19.neventEncode(event)

              response_link.href = response_link.innerText = "https://coracle.social/" + nevent
            }

            socket.close()
          }
        }catch(e){
          document.querySelector("#response").innerText = e.message
        }
      }
    </script>
  </body>
</html>


#publishevent

just publish json events with one click

this tool handles signing for you

(here is the correct link uri)

data:text/html;base64,<!DOCTYPE html>
<html lang="">
  <head>
    <meta charset="utf-8">
    <title>publishevent</title>
    <style>
    textarea {
      width: 50%;
      height: 30em;
      box-sizing: border-box;
      tab-size : 2;
      resize: vertical;
    }
    #response {
      float: right;
      width: calc( 50% - 2em );
      padding: 0 1em;
      word-break: break-word;
    }
    </style>
    <script src="https://unpkg.com/nostr-tools/lib/nostr.bundle.js" crossorigin="anonymous"></script>
    <script>
    (function (exports) {
      'use strict';

      function insertTextIntoField(field, text) {
          const document = field.ownerDocument;
          const initialFocus = document.activeElement;
          if (initialFocus !== field) {
              field.focus();
          }
          if (text === '') {
              document.execCommand('delete');
          }
          else {
              document.execCommand('insertText', false, text);
          }
          if (initialFocus === document.body) {
              field.blur();
          }
          else if (initialFocus instanceof HTMLElement && initialFocus !== field) {
              initialFocus.focus();
          }
      }

      function indentSelection(element) {
          const { selectionStart, selectionEnd, value } = element;
          const selectedText = value.slice(selectionStart, selectionEnd);
          const lineBreakCount = /\n/g.exec(selectedText)?.length;
          if (lineBreakCount > 0) {
              const firstLineStart = value.lastIndexOf('\n', selectionStart - 1) + 1;
              const newSelection = element.value.slice(firstLineStart, selectionEnd - 1);
              const indentedText = newSelection.replaceAll(/^|\n/g,
              '$&\t');
              const replacementsCount = indentedText.length - newSelection.length;
              element.setSelectionRange(firstLineStart, selectionEnd - 1);
              insertTextIntoField(element, indentedText);
              element.setSelectionRange(selectionStart + 1, selectionEnd + replacementsCount);
          }
          else {
              insertTextIntoField(element, '\t');
          }
      }
      function findLineEnd(value, currentEnd) {
          const lastLineStart = value.lastIndexOf('\n', currentEnd - 1) + 1;
          if (value.charAt(lastLineStart) !== '\t') {
              return currentEnd;
          }
          return lastLineStart + 1;
      }
      function unindentSelection(element) {
          const { selectionStart, selectionEnd, value } = element;
          const firstLineStart = value.lastIndexOf('\n', selectionStart - 1) + 1;
          const minimumSelectionEnd = findLineEnd(value, selectionEnd);
          const newSelection = element.value.slice(firstLineStart, minimumSelectionEnd);
          const indentedText = newSelection.replaceAll(/(^|\n)(\t| {1,2})/g, '$1');
          const replacementsCount = newSelection.length - indentedText.length;
          element.setSelectionRange(firstLineStart, minimumSelectionEnd);
          insertTextIntoField(element, indentedText);
          const firstLineIndentation = /\t| {1,2}/.exec(value.slice(firstLineStart, selectionStart));
          const difference = firstLineIndentation
              ? firstLineIndentation[0].length
              : 0;
          const newSelectionStart = selectionStart - difference;
          element.setSelectionRange(selectionStart - difference, Math.max(newSelectionStart, selectionEnd - replacementsCount));
      }
      function tabToIndentListener(event) {
          if (event.defaultPrevented
              || event.metaKey
              || event.altKey
              || event.ctrlKey) {
              return;
          }
          const textarea = event.target;
          if (event.key === 'Tab') {
              if (event.shiftKey) {
                  unindentSelection(textarea);
              }
              else {
                  indentSelection(textarea);
              }
              event.preventDefault();
              event.stopImmediatePropagation();
          }
          else if (event.key === 'Escape'
              && !event.shiftKey) {
              textarea.blur();
              event.preventDefault();
              event.stopImmediatePropagation();
          }
      }
      function enableTabToIndent(elements, signal) {
          if (typeof elements === 'string') {
              elements = document.querySelectorAll(elements);
          }
          else if (elements instanceof HTMLTextAreaElement) {
              elements = [elements];
          }
          for (const element of elements) {
              element.addEventListener('keydown', tabToIndentListener, { signal });
          }
      }
      const indent = indentSelection;
      const unindent = unindentSelection;
      const eventHandler = tabToIndentListener;
      const watch = enableTabToIndent;

      exports.enableTabToIndent = enableTabToIndent;
      exports.eventHandler = eventHandler;
      exports.indent = indent;
      exports.indentSelection = indentSelection;
      exports.tabToIndentListener = tabToIndentListener;
      exports.unindent = unindent;
      exports.unindentSelection = unindentSelection;
      exports.watch = watch;

      Object.defineProperty(exports, '__esModule', { value: true });
    }(this.window = this.window || {}));
    </script>
  </head>
  <body>
    <form action="#">
      <div id="response"></div>
      <p>
        <textarea></textarea>
      </p>
      <p>
        apply template:
        <a href="#" data-template="1">kind 1</a>,
        <a href="#" data-template="1063">kind 1063</a>
      </p>
      <p>
        <input type="submit" value="publish"/>
      </p>
    </form>
    <script>
      const templates = {}

      templates[1] = {
        "kind": 1,
        "content": "",
        "tags": []
      }

      templates[1063] = {
        "kind": 1063,
        "tags": [
          ["url",""]
        ],
        "content": "",
      }

      for(let key in templates){
        templates[key] = JSON.stringify(templates[key], null, "\t")
      }

      const textarea = document.querySelector("textarea")

      function apply_template(link){
        textarea.value = templates[link.dataset.template]
      }

      for(let template_link of document.querySelectorAll("a[data-template]")){
        template_link.onclick = function(e){
          e.preventDefault()
          apply_template(this)
        }
      }

      apply_template(document.querySelector("a[data-template]"))
      enableTabToIndent(textarea)

      document.querySelector("form").onsubmit = function(e){
        e.preventDefault()

        try{
          const event = JSON.parse(textarea.value)
          event.created_at = event.created_at || Math.floor(new Date().getTime() / 1000)

          const sk = NostrTools.generateSecretKey()
          NostrTools.finalizeEvent(event, sk)

          const socket = new WebSocket("wss://nos.lol")

          socket.onopen = function(){
            socket.send(JSON.stringify(["EVENT", event]))
          }

          socket.onmessage = function(e){
            document.querySelector("#response").innerText = e.data
            socket.close()
          }
        }catch(e){
          document.querySelector("#response").innerText = e.message
        }
      }
    </script>
  </body>
</html>


#publishevent

just publish json events with one click

this tool handles signing for you

data:text/html;base64,<!DOCTYPE html>
<html>
<head>
    <title>filepublish</title>
    <meta charset="UTF-8"/>
    <meta name="viewport" content="width=device-width, user-scalable=no"/>
    <style>
      form {
        margin-bottom: 1em;
        width: calc(100% - 1em);
        max-width: 30em;
      }
      
      form > * {
        display: block;
        margin: .5em 0;
      }
      
      .right {
        float: right;
      }
      
      textarea {
        width: 100%;
        box-sizing: border-box;
        height: 10em;
      }
    </style>
  </head>
  <body>
    <strong>nostr file uploader</strong>

    <form method="post" enctype="multipart/form-data" id="note-form" action="#">
        <input type="text" id="tags" placeholder="list of tags"/>
        <textarea id="comment" placeholder="comment"></textarea>
        <div class="inline">
          <input type="file" class="file" name="file" multiple=""/>
          <div class="right">
            type 
            <select id="type">
              <option>file upload</option>
              <option>text</option>
            </select>
          </div>
        </div>
        <input type="submit"/>
    </form>

    <div>initializing</div>
    <div>scroll <a data-action="manuscroll" href="#">manual</a>, <a data-action="autoscroll" href="#">auto</a></div>

    <script src="https://unpkg.com/nostr-tools/lib/nostr.bundle.js" crossorigin="anonymous"></script>
    <script src="https://unpkg.com/crypto-js@4.1.1/crypto-js.js" integrity="sha256-8L3yX9qPmvWSDIIHB3WGTH4RZusxVA0DDmuAo4LjnOE=" crossorigin="anonymous"></script>
    <script> 
      window.loaded = true
      let files = []
      let socket = null
      let reshandlers = []
      const splitpartsize = 200
      let q = location.hash.substring(1).trim()
      let options = q.match(/\/[^\/]+/g) || []
      const privk = NostrTools.generateSecretKey()
      const qstr = location.search.match(/fileloader=([a-z0-9]+)/i)
      
      const fileloaderurl = qstr && qstr[1] + "?filepublish=" + location.pathname.match(/[a-z0-9]+$/i) || "fileloader.html"
      const allrelays = [
      	//"wss://n.xmr.se",
        "wss://nos.lol", 
        "wss://relay.nostr.band",
        "wss://eden.nostr.land/",
		    "wss://relayable.org",
	      "wss://nostr.mom",
		    "wss://relay.nostr.band",
	      "wss://nostr.self-determined.de",
		    "wss://puravida.nostr.land"
      ]
      
      info('browse files with <a href="' + fileloaderurl + '">fileloader.html</a>', true)

      for(let relay of allrelays){
        info('click to connect to <a data-action="relay" data-value="' + relay + '" href="#">' + relay + '</a>', true)
      }

      let relay = getoption("relay")[1] && decodeURIComponent(getoption("relay")[1]) || allrelays[0]

      function delete_empty(arr){
        for(let key in arr){
          if(arr[key] == null || arr[key].length == 0){
            delete arr[key]
          }
        }
      }

      async function publishfile(filename, comment, fileids, partindex, partcount){
        const res = await publishevents(335, [JSON.stringify({
          name: filename, 
          comment: comment, 
          fileids: fileids, 
          part: partindex,
          partcount: partcount
        })], () => {})
        
        return res[0].id
      }

      function readfile(file){
        return new Promise((resolve, reject) => {
          let reader = new FileReader()
          reader.readAsDataURL(file)
          
          reader.onload = async function(readerEvent){
            resolve(readerEvent.target.result)
          }
        })
      }

      function base64encode(content){
        return new Promise(resolve => {
          const bytes = new TextEncoder().encode(content)
          const blob = new Blob([bytes], { type: files[0].type })
          const reader = new FileReader()

          reader.onload = () => resolve(reader.result)

          reader.readAsDataURL(blob)
        })
      }

      async function uploadfile(data, filename, filetype, comment, progresscb){
        let content = data.match(/data:[a-z0-9]{1,20}\/[a-z0-9\-\.]{1,20};base64,(\S+)/)[1]
        
        if(filetype == "video/webm" && !confirm("is this a video file?")){
          filetype = "audio/webm"
          info("content type set to " + filetype)
        }
        
        let contentres = preparefile(content, filetype)

        try{
          let res = await publishevents(contentres.kind, contentres.contents, progresscb)

          if(res.length == 0){
            return
          }

          let fileids = Array.from(new Set(res.map(e => e.id)))
          let file_event_ids = []
          let i = 0
          const partcount = Math.ceil(fileids.length / splitpartsize)

          for(; i < partcount; i++){
            start = i * splitpartsize
            file_event_ids.push(await publishfile(filename, comment, fileids.slice(start, start+splitpartsize), i, partcount))     
          }
          
          let fileres = await publishevents(336, [JSON.stringify({
            name: filename, 
            comment: comment, 
            ids: file_event_ids
          })], progresscb)
          
          return fileres[0].id
        }catch(e){
          info(e)
        }
      }
      
      async function pubkey(){
        return window.nostr && await window.nostr.getPublicKey() || NostrTools.getPublicKey(privk)
      }
      
      async function sign(event){
        if(window.nostr){
          return await nostr.signEvent(event)
        }

        event = NostrTools.finalizeEvent(event, privk)
        
        return event
      }

      async function prepare_event(event){
        event.pubkey = await pubkey()
        event.created_at = parseInt(Date.now() / 1000, 10)
        event.id = NostrTools.getEventHash(event)
        return event
      }
          
      async function publish(event){
        event = await prepare_event(event)
        let signedevent = await sign(event)
        let json = JSON.stringify(["EVENT", signedevent])
        socket.send(json)
        
        return event
      }

      function tags(){
        let tags_str = (document.querySelector("#tags").value.replaceAll("#", "").replace(/ +/g, " "))
        return tags_str.length == 0 && [] || tags_str.split(" ").map(tag => ["t", tag])
      }

      async function publishevents(kind, contents, progresscb){
        const events = []
        
        for(let i = 0; i < contents.length; i++){
          let content = contents[i]
          let relayres = null
          let res = null
          let error = null
          
          try{
            res = await publish({
              "kind": kind,
              "content": content,
              "tags": tags()
            })
          
            relayres = await waitres(null, "OK")
            
            if(relayres[1][2] == false){
              error = "error " + relayres[1][3]
            }
          }catch(e){
            error = "error " + e
          }
          
          if(error){
            info(error)
            const nextrelay = allrelays.find(r => r != relay)
            
            if(confirm(error + ". resume with " + nextrelay +  "?")){
              i--
              relay = nextrelay
              await connectrelays()
              console.log("continue upload")
              continue
            }
            
            return events
          }

          progresscb(parseInt(i / contents.length * 100, 10))
          events.push(res)
        }
 
        return events
      }

      function updatevalueoption(key, value){
        filterstr = value || prompt(key, getoption(key)[1] || "") || ""
        updatehash(filterstr.length > 0 && "/" + key + ":" + filterstr || null, ["/" + key + ":" + (getoption(key)[1] || "")])
      }
      
      function getoption(key){
        let filteropt = options.find(o => o.indexOf("/" + key + ":") == 0)
        let matcher = new RegExp('^\/' + key + '\\:(\\S+)$')
        let match = filteropt && filteropt.match(matcher) || null
        return match || []
      }
      
      function preparefile(content, filetype){
        const uploadpartsize = 55000
 
        let contents = []
        let partcount = Math.ceil(content.length / uploadpartsize)
        const hash = CryptoJS.SHA1(content).toString()
        
        for(let i = 0; i < partcount; i++){
          let part = content.substr(i * uploadpartsize, uploadpartsize)
          
          if(part.length > 0){
            contents.push((partcount > 1 && "part" + (i + 1) + ";" + partcount + ";" + hash + ";" || "") + (i == 0 && filetype + ";base64," || "") + part)
          }
        }

        return {
          kind: 334, 
          contents: contents
        }
      }

      info("ready")

      addEventListener("change", async (e) => {
        if(e.target.type == "file"){
          files = e.target.files
        }
      })

      addEventListener("submit", async e => {
        e.preventDefault()
        let data = null
        let comment = ""
        
        if(document.querySelector("#type").value == "text"){
          let content = document.querySelector("#comment").value
          
          files = [
            {
              name: "text.txt", 
              type: "text/plain", 
              size: content.length
            }
          ]
          
          data = await base64encode(content, files[0].type)
        }
        
        if(files.length == 0){
          info("no file selected")
          return
        }

        let totalsize = 0
        
        for(let file of files){
          totalsize += file.size
        }
        
        info("uploading", files.length)

        let fileids = []
        
        for(let i = 0; i < files.length; i++){
          let file = files[i]
          
          if(!data){
            data = await readfile(file)
            comment = document.querySelector("#comment").value.trim()
          }
          
          const fileid = await uploadfile(data, file.name, file.type || "text/plain", comment, (progress) => {
            info("uploading " + files.length + 
              " files (" + (i + 1) + " / " + files.length + " | " + progress + " %)")
          })

          if(fileid == null){
            continue
          }

          fileids.push(fileid)
        }

        info('File(s) uploaded: ' + fileids.length)

        for(let fileid of fileids){
          const url = fileloaderurl + "#" + encodeURIComponent(fileid) + "/inline/quiet"
          info('fileurl: <a href="' + url + '">' + url + '</a>', true)

          if(options.includes("/reply")){
            const tagurl = fileloaderurl + "#/inlineall/quiet/tag:" + tags()[0][1]
            info('tagurl: <a href="' + tagurl + '">' + tagurl + '</a>', true)
          }
        }
        
        if(fileids.length > 0){
          document.querySelector("form").reset()
        }
      })

      function info(text, html, important, className, file, fileid){
        let scrollmax = document.scrollingElement.scrollTop == document.scrollingElement.scrollTopMax
        let div = document.createElement("div")
        div.classList.add(className)
        div[html && "innerHTML" || "innerText"] = text
        document.body.append(div)
        div.dataset.fileid = fileid
        
        if(file){
          div.dataset.filename = file.name
          div.dataset.fileobj = JSON.stringify(file)
        }
        
        if(socket){
          div.dataset.socketurl = socket.url
        }
        
        if(options.includes("/autoscroll") && scrollmax){
          document.scrollingElement.scrollTop = document.scrollingElement.scrollTopMax
        }
      }

      function waitres(key, type){
        let reshandler = function(resolve, reject){
          this.type = type || "EOSE"
          this.events = []
          this.handle = function(res){
            let o = JSON.parse(res.data)
            if(key == null || o[1] == key){
              if(o[0] == this.type){
                resolve([this.events, o])
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }
              if(o[0] == "NOTICE"){
                reject(o[1])
                reshandlers.splice(reshandlers.indexOf(this), 1)
                return
              }

              this.events.push(o[2])
            }
          }
        }
        
        return new Promise((resolve, reject) => {
          let handler = new reshandler(resolve, reject)
          reshandlers.push(handler)
        })
      }
      
      function connect(url){
        let socket = new WebSocket(url)
        socket.onmessage = message => reshandlers.forEach(rh => rh.handle(message))
        return socket
      }
      
      function connectrelays(){
        return new Promise((resolve, reject) => {
          if(socket){
            socket.close()
          }
          
          socket = connect(relay)
          
          socket.addEventListener("open", function(){
            info("connected to " + socket.url)
            resolve()
          })
        })
      }
      
      function updatehash(add, remove){
        if(remove != null && !Array.isArray(remove)){
          throw new Error("updatehash, remove should be array")
        }
        
        for(let r of (remove || []).sort((a, b) => a.length < b.length)){
          location.hash = location.hash.replace(r, "")
        }
        
        location.hash += location.hash.indexOf(add) == -1 && add || ""
      }

      function setrelay(relay){
        updatehash("/relay:" + encodeURIComponent(relay), [getoption("relay")[0]])
      }

      addEventListener("click", function(e){
        if(e.target.dataset.action){
          const c = {
            relay: () => setrelay(e.target.dataset.value),
            manuscroll: () => updatehash(null, ["/autoscroll"]),
            autoscroll: () => updatehash("/autoscroll", null), 
          }
          
          e.preventDefault()
          c[e.target.dataset.action]()
        }
      })
      
      addEventListener("hashchange", function(e){
        location.reload()
      })

      addEventListener("load", async () => {
        document.querySelector(".file").value = null
        document.querySelector("#type").value = document.querySelector("#type option").value
        document.querySelector("#tags").value = getoption("tags")[1] || ""
        connectrelays()
      })
    </script>
</body>
</html>


it loads initial code from this data uri, all the rest is loaded from nostr, so it doesnt rely on any single http server.

#dui #datauri_info #dataurinfo #datauri #info

(updates: added #video200)

data:text/html;r=nos.lol;k=2b8d50fcc974ffb652c0ab33edebb6a9ce72cd154591969592bc8b607a4ea1f2;f=datauri;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K

every agent is io-machine

how ai takes over the world might not happen as you think it will

its simply that majority of data processing happens within virtual neural networks

machines will take over the world by doing more input output processing within their neural networks

basically every neural network agent is just io-machine fine tuned to its environment or by its environment

natural agents fine tune automatically by the feedback from environment

its like water flowing. every naturally learning agent goes the way of least resistance

#llm #ai #neuralnetworks #learning #agi

who will be programming who? are you programming machine or is machine programming you?

natural language is a programming language for human mind

and llms are excellent at producing it

think

United States of usA

language recursion

idk

just interesting for some reason

#wired

data:text/html;r=nos.lol;k=2b8d50fcc974ffb652c0ab33edebb6a9ce72cd154591969592bc8b607a4ea1f2;f=w;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K#board/npub19nrn4l0s39kpwww7pgk9jddj8lzekqxmtrll8r2a57chtq3zx6sq00vetn

Replying to Avatar b

data:text/html;base64,<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <title>datauri info</title>
    <style>
      body {
        margin: 1em 1em 3em 1em;
      }

      header h1,
      header h2 {
        display: inline-block;
        margin-right: 1em;
      }

      .menu,
      .content {
        display: none;
      }

      .active {
        display: block;
      }

      h2 > a {
        padding: .2em .3em;
      }

      a, a:visited {
        color: inherit;
      }

      a:hover {
        color: rgb(200, 0, 0);
      }

      .data {
        background: rgb(220, 220, 100);
        /*background: rgb(180, 200, 230);*/
        /*background: rgb(180, 200, 230);*/
        overflow: hidden;
        word-break: break-all;
        max-height: 10em;
        padding: .3em;
      }

      .data,
      a {
        border-radius: .3em;
      }

      .data.scroll {
        overflow: scroll;
      }

      h2 {
        margin: 1em 0 .4em 0;
      }

      h2:first-child {
        margin-top: 0;
      }

      p {
        margin: 0 0 1em 0;
      }
    </style>
  </head>
  <body>
    <header>
      <div class="menu active">
        <h1>datauri info</h1>
        <h2>
          <a href="#devs">datauri for devs</a>
        </h2>
      </div>
      <div class="menu">
        <h1>datauri for devs</h1>
        <h2>
          <a href="#">datauri info</a>
        </h2>
      </div>
    </header>
    <main>
      <div class="content active">
        <p>
        What is datauri application? Apps hosted as datauris are basically portable web apps that work without http server.
        If you see this page, you probably already know how datauri apps work. You just copy-paste the link and access the app.
        Here are some datauri apps.
        </p>
        <h2>wired</h2>
        <p>Anon approved pow based nostr client</p>
        <p class="data">data:text/html;base64,PCFkb2N0eXBlIGh0bWw+PGh0bWwgbGFuZz0iZW4iPjxoZWFkPjxtZXRhIGNoYXJzZXQ9InV0Zi04Ii8+PG1ldGEgbmFtZT0idmlld3BvcnQiIGNvbnRlbnQ9IndpZHRoPWRldmljZS13aWR0aCxpbml0aWFsLXNjYWxlPTEiLz48bWV0YSBuYW1lPSJ0aGVtZS1jb2xvciIgY29udGVudD0iIzAwMDAwMCIvPjxtZXRhIG5hbWU9Im1vYmlsZS13ZWItYXBwLWNhcGFibGUiIGNvbnRlbnQ9InllcyIvPjxtZXRhIG5hbWU9ImFwcGxlLXRvdWNoLWZ1bGxzY3JlZW4iIGNvbnRlbnQ9InllcyIvPjxtZXRhIG5hbWU9ImFwcGxlLW1vYmlsZS13ZWItYXBwLXRpdGxlIiBjb250ZW50PSJFeHBvIi8+PG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+PG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtc3RhdHVzLWJhci1zdHlsZSIgY29udGVudD0iZGVmYXVsdCIvPjxtZXRhIG5hbWU9ImRlc2NyaXB0aW9uIiBjb250ZW50PSJUaGUgV2lyZWQiLz48dGl0bGU+VGhlIFdpcmVkPC90aXRsZT48L2hlYWQ+PGJvZHk+PG5vc2NyaXB0PllvdSBuZWVkIHRvIGVuYWJsZSBKYXZhU2NyaXB0IHRvIHJ1biB0aGlzIGFwcC48L25vc2NyaXB0PjxkaXYgaWQ9InJvb3QiPjwvZGl2PjxzY3JpcHQ+Cihhc3luYyBmdW5jdGlvbigpewogIGFzeW5jIGZ1bmN0aW9uIGRnZXQoaWQsIHJlbGF5KXsKICAgIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSkgPT4gewogICAgICBjb25zdCBzb2NrZXQgPSBuZXcgV2ViU29ja2V0KHJlbGF5KQogICAgICBsZXQgZmlsZV9kYXRhID0gW10KICAgICAgbGV0IGZpbGVpZHMgPSBbXQoKICAgICAgc29ja2V0Lm9ub3BlbiA9IGZ1bmN0aW9uKCl7CiAgICAgICAgc29ja2V0LnNlbmQoSlNPTi5zdHJpbmdpZnkoWyJSRVEiLCAiZmlsZWlkcyIsIHsiaWRzIjogW2lkXX1dKSkKICAgICAgfQoKICAgICAgc29ja2V0Lm9ubWVzc2FnZSA9IGFzeW5jIGZ1bmN0aW9uKGUpewogICAgICAgIGNvbnN0IGRhdGEgPSBKU09OLnBhcnNlKGUuZGF0YSkKCiAgICAgICAgaWYoZGF0YVswXSA9PT0gIkVWRU5UIiAmJiBkYXRhWzFdID09PSAiZmlsZWlkcyIpewogICAgICAgICAgZmlsZWlkcy5wdXNoKC4uLkpTT04ucGFyc2UoZGF0YVsyXS5jb250ZW50KS5maWxlaWRzKQogICAgICAgICAgc29ja2V0LnNlbmQoSlNPTi5zdHJpbmdpZnkoWyJSRVEiLCAicGFydHMiLCB7ImlkcyI6IGZpbGVpZHMsICJsaW1pdCI6IDMwMDB9XSkpCiAgICAgICAgfQogICAgICAgIGVsc2UgaWYoZGF0YVswXSA9PT0gIkVWRU5UIiAmJiBkYXRhWzFdID09PSAicGFydHMiKXsKICAgICAgICAgIGNvbnN0IG1hdGNoID0gZGF0YVsyXS5jb250ZW50Lm1hdGNoKC9ecGFydChcZHsxLDR9KTsoXGR7MSw0fSk7W2EtejAtOV0rOy8pCgogICAgICAgICAgaWYobWF0Y2gpewogICAgICAgICAgICBjb25zdCBwYXJ0X2RhdGEgPSBkYXRhWzJdLmNvbnRlbnQuc3Vic3RyKG1hdGNoWzBdLmxlbmd0aCkKICAgICAgICAgICAgZmlsZV9kYXRhLnB1c2goW3BhcnNlSW50KG1hdGNoWzFdLCAxMCksIHBhcnRfZGF0YV0pCiAgICAgICAgICB9ZWxzZXsKICAgICAgICAgICAgZmlsZV9kYXRhLnB1c2goWzEsIGRhdGFbMl0uY29udGVudF0pCiAgICAgICAgICB9CgogICAgICAgICAgaWYoZmlsZV9kYXRhLmxlbmd0aCA9PT0gZmlsZWlkcy5sZW5ndGgpewogICAgICAgICAgICBmaWxlX2RhdGEgPSBmaWxlX2RhdGEuc29ydCgoYSwgYikgPT4gYVswXSAtIGJbMF0pLm1hcChlID0+IGVbMV0pCiAgICAgICAgICAgIGNvbnN0IGRhdGF1cmkgPSAiZGF0YToiICsgZmlsZV9kYXRhLmpvaW4oIiIpCiAgICAgICAgICAgIGNvbnN0IGJsb2IgPSBhd2FpdCAoYXdhaXQgZmV0Y2goZGF0YXVyaSkpLmJsb2IoKQogICAgICAgICAgICByZXNvbHZlKGJsb2IpCiAgICAgICAgICB9CiAgICAgICAgfQogICAgICB9CiAgICB9KQogIH0KCiAgY29uc3Qgd29ya2VyX3NvdXJjZSA9IGF3YWl0IGRnZXQoImZlZWNlYTAxZmU2ZDkyMGU5ZjU2M2U5OWU1MjdhMmE1MjVlODQwOTUyN2ViMjc3Yzk0ZDJlOTA3MzRkYzRjNTAiLCAid3NzOi8vbm9zLmxvbCIpCiAgY29uc3Qgd29ya2VyXzU5Ml91cmwgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGF3YWl0IGRnZXQoImZlMGVhNjA2ODBiMTYwYjYyN2M0YmYzNThlMDk1YWZhMTcxNWJhMTlhNWI5YzkwOTYzMGQ1MmQzZTQzMjQ3MmMiLCAid3NzOi8vbm9zLmxvbCIpKQogIGNvbnN0IGljb25fcG5nX3VybCA9IFVSTC5jcmVhdGVPYmplY3RVUkwoYXdhaXQgZGdldCgiMDVkZTM0ZWE5MDkyMmVlOTQxMDUzYzQzYjUwMGRhYTE0ODRlMWY3MzM2M2ZjNWRhNzIyMzRiNDAzYjk3NzM3OCIsICJ3c3M6Ly9ub3MubG9sIikpCiAgY29uc3QgbWFpbl9qc19tYXBfdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChhd2FpdCBkZ2V0KCI1NWVkNDdiMjk0NzJkZTE4Y2I3MWQ0M2NjOGNmZmU3NTcwNzJmNDA3Y2ZlOGNlYjRkMDQyNDdiZGU0NDc1M2VlIiwgIndzczovL25vcy5sb2wiKSkKCiAgbGV0IHdvcmtlcl9zb3VyY2Vfc3RyID0gYXdhaXQgd29ya2VyX3NvdXJjZS50ZXh0KCkKICB3b3JrZXJfc291cmNlX3N0ciA9IHdvcmtlcl9zb3VyY2Vfc3RyLnJlcGxhY2UoJ2ltcG9ydFNjcmlwdHModC5wK3QudShyKSknLCAnaW1wb3J0U2NyaXB0cygiJyArIHdvcmtlcl81OTJfdXJsICsgJyIpJykKICBjb25zdCB3b3JrZXJfdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChuZXcgQmxvYihbd29ya2VyX3NvdXJjZV9zdHJdLCB7dHlwZTogInRleHQvamF2YXNjcmlwdCJ9KSkKCiAgY29uc3Qgb3JpZ2luYWxXb3JrZXJDb25zdHJ1Y3RvciA9IFdvcmtlcjsKCiAgV29ya2VyID0gZnVuY3Rpb24odXJsKSB7CiAgICB1cmwgPSB3b3JrZXJfdXJsCiAgICByZXR1cm4gbmV3IG9yaWdpbmFsV29ya2VyQ29uc3RydWN0b3IodXJsKTsKICB9CgogIFVSTCA9IGNsYXNzIE15VVJMIGV4dGVuZHMgVVJMIHsKICAgIGNvbnN0cnVjdG9yKGlucHV0KSB7CiAgICAgIHN1cGVyKGlucHV0LmluZGV4T2YoIi8iKSA9PT0gMCAmJiAobG9jYXRpb24ucHJvdG9jb2wgKyBsb2NhdGlvbi5wYXRobmFtZSArIGlucHV0KSB8fCBpbnB1dCk7CiAgICB9CiAgfQoKICBkYXRhTG9jYWxTdG9yYWdlID0gewogICAgZ2V0SXRlbTogZnVuY3Rpb24oKXt9LAogICAgc2V0SXRlbTogZnVuY3Rpb24oKXt9CiAgfQoKICBjb25zdCBzY3JpcHRzID0gWwogICAgIjE1NDNhMDcwYTQyZThlMGMwNTg2MTg0MGJkZTdlNWI2OTFjNGI1ZGVjNjYyNzQ1ODUyYzIyODNjZmVmZDhjYTUiCiAgXQoKICBjb25zdCBzdHlsZXMgPSBbCiAgICAiNjk5NjA3NzRjZTZkZmE4MzFhMzMwODczNDEyNWQ4MGRlMjY5YTg0ZjU5ODg4YTdkMzEyMTVlYTFhYzU1MTE1MiIKICBdCgogIGNvbnN0IHNjcmlwdF9pZCA9IHNjcmlwdHNbMF0KICBjb25zdCBzY3JpcHRfc291cmNlID0gYXdhaXQgZGdldChzY3JpcHRfaWQsICJ3c3M6Ly9ub3MubG9sIikKICBsZXQgc2NyaXB0X3NvdXJjZV9zdHIgPSBhd2FpdCBzY3JpcHRfc291cmNlLnRleHQoKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZSgiL2ljb24ucG5nIiwgaWNvbl9wbmdfdXJsKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZSgvKHNvdXJjZU1hcHBpbmdVUkw9KShtYWluXC5bYS16MC05XXs4fVwuanNcLm1hcCkvLCAiJDFodHRwOi8vbG9jYWxob3N0OjgwMDAvJDIiKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZUFsbCgibG9jYWxTdG9yYWdlIiwgImRhdGFMb2NhbFN0b3JhZ2UiKQoKICBjb25zdCBzY3JpcHRfZWwgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCJzY3JpcHQiKQogIHNjcmlwdF9lbC5zcmMgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKG5ldyBCbG9iKFtzY3JpcHRfc291cmNlX3N0cl0sIHt0eXBlOiAidGV4dC9qYXZhc2NyaXB0In0pKQogIGNvbnNvbGUubG9nKCJzY3JpcHRfZWwuc3JjIiwgc2NyaXB0X2VsLnNyYykKICBkb2N1bWVudC5ib2R5LmFwcGVuZChzY3JpcHRfZWwpCgogIGZvcihsZXQgc3R5bGVfaWQgb2Ygc3R5bGVzKXsKICAgIGNvbnN0IHN0eWxlX3NvdXJjZSA9IGF3YWl0IGRnZXQoc3R5bGVfaWQsICJ3c3M6Ly9ub3MubG9sIikKICAgIGNvbnN0IHN0eWxlX2VsID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgibGluayIpCiAgICBzdHlsZV9lbC5yZWwgPSAic3R5bGVzaGVldCIKICAgIHN0eWxlX2VsLmhyZWYgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGF3YWl0IHN0eWxlX3NvdXJjZSkKICAgIGRvY3VtZW50LmJvZHkuYXBwZW5kKHN0eWxlX2VsKQogIH0KfSkoKQo8L3NjcmlwdD4KPC9ib2R5PjwvaHRtbD4K</p>

        <h2>ourchan</h2>
        <p>Anon approved image board</p>
        <p class="data">data:text/html;r=nos.lol;k=2b8d50fcc974ffb652c0ab33edebb6a9ce72cd154591969592bc8b607a4ea1f2;f=ourchan;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>filepublish</h2>
        <p>Upload files</p>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=filepublish.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>fileloader</h2>
        <p>Download/view files</p>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=fileloader.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>video clip creator</h2>
        <p>Create and upload 1 second video clips</p>
        <p class="data">data:text/html;base64,<!DOCTYPE html>
<html>
  <head>
    <title>video clip creator</title>
    <style>
    body {
      margin: 1em;
    }
    h1 {
      margin-top: 0;
      font-size: 1.5em;
    }
    #log {
      overflow: auto;
      position: absolute;
      bottom: 1em;
      top: 11em;
      border: .3em solid #666;
      border-radius: .3em;
      right: 1em;
      left: 1em;
      padding: .5em;
      background: #eee;
    }
    #progress {
      background: #ddd;
      border-radius: .3em;
    }
    #progressbar {
      height: 1.3em;
      background: blue;
      width: 0%;
      border-radius: .4em;
    }
    #log > a {
      margin-bottom: 1em;
      display: block;
    }
    p {
      margin-top: 0;
      max-height: 7em;
      overflow: scroll;
    }
    video {
      display: none;
      z-index: 1;
    }
    #canvas,
    video {
      position: absolute;
      top: 1em;
      right: 1em;
      width: 10em;
      height: 6em;
    }
    </style>
  </head>
  <body>
    <h1>video clip creator</h1>
    <form id="form">
      <p>
        <input id="file" type="file"/>
      </p>
      <p>
        <input id="submit" type="submit" value="encode"/>
      </p>
    </form>
    <video id="video"></video>
    <video id="video2" controls loop="1"></video>
    <canvas id="canvas"></canvas>
    <div id="progress">
      <div id="progressbar"></div>
    </div>
    <div id="log"><div>
    <script type="module">
    const canvas = document.getElementById("canvas")
    const ctx = canvas.getContext('2d')
    const log = document.getElementById("log")
    const video = document.getElementById("video")
    const video2 = document.getElementById("video2")
    const progressbar = document.getElementById("progressbar")
    let files = null

    const frame_duration = 1/30
    const extract_frame_count = 30//1 / frame_duration
    let time = 15

    const buffer = await fetch(
      "https://unpkg.com/webm-wasm@0.4.1/dist/webm-worker.js"
    ).then(r => r.arrayBuffer())

    const worker = new Worker(
      URL.createObjectURL(new Blob([buffer], { type: "text/javascript" }))
    )

    let encoded_data = []
    let start_time = null
    let ready = false

    document.getElementById("form").onsubmit = function(e){
      e.preventDefault()
      video2.style.display = "none";
      video2.pause()
      progressbar.style.width = "0.5%"
      this.querySelectorAll("input").forEach(el => el.disabled = true)
      document.getElementById("submit").value = "encoding"
      files = document.getElementById("file").files
      start()
    }

    function blobToDataURL(blob, callback) {
      return new Promise(resolve => {
        var a = new FileReader()
        a.onload = function(e) {
          resolve(e.target.result)
        }
        a.readAsDataURL(blob)
      })
    }

    function info(msg, type){
      const el = document.createElement(type === 2 && "a" || "p")
      el[(type === 3) && "innerHTML" || "innerText"] = msg

      if(type){
        el.href = msg
      }

      log.append(el)
      log.scrollTo(0, 1e6)
    }

    async function process_video(){
      let frame_count = 0

      video.onseeked = function(e) {
        frame_count++
        info("extract frame " + frame_count + " / " + extract_frame_count)

        ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
        const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height)

        if(frame_count < extract_frame_count){
          time += frame_duration
          this.currentTime = Math.min(Math.max(0, (time < 0 ? this.duration : 0) + time), this.duration)
          worker.postMessage(imageData.data.buffer, [imageData.data.buffer])
        }else{
          worker.postMessage(null)
        }
      }
    }

    function total_time(start){
      return (new Date().getTime() - start) / 1000
    }

    function createBufferURL(buffer, type = '') {
      return URL.createObjectURL(new Blob([buffer], {type}));
    }

    async function show_data_uri(blob){
      let blob_url = URL.createObjectURL(blob)

      info(await blobToDataURL(blob))
      info(blob_url, 2)

      video2.src = blob_url
      video2.style.display = "block";
      video2.play()

      progressbar.style.width = "100%"
      document.getElementById("form").reset()
      document.getElementById("submit").value = "encode"
      document.querySelectorAll("form input").forEach(el => el.disabled = false)
    }

    worker.onmessage = function(e){
      if(!e.data) {
        return
      }

      if(e.data != null && typeof e.data == "object" && e.data.byteLength > 0){
        encoded_data.push(e.data)
        info("encode " + encoded_data.length + " / " + extract_frame_count)
        progressbar.style.width = parseInt((encoded_data.length / extract_frame_count) * 100, 10) + "%"
      }

      if(e.data == "READY"){
        info("ready", canvas.width, canvas.height)

        worker.postMessage({
          width: canvas.width,
          height: canvas.height,
          realtime: true,
          bitrate: 100
        })

        //for(let i = 0; i < frames.length; i++){
        //  worker.postMessage(frames[i], [frames[i]])
        //}

        ready = true
        process_video()
      }
      else if(ready){
        if(e.data.byteLength == 0){
          info("encoding finished (duration: " + parseInt(total_time(start_time), 10) +
            " s, speed: " + parseInt(total_time(start_time) / extract_frame_count, 10) +" s / frame)")
          let blob = new Blob(encoded_data, { type: 'video/webm' })
          show_data_uri(blob)
        }
      }
    }

    function load_video(){
      return new Promise(resolve => {
        video.src = URL.createObjectURL(files[0])

        video.onloadedmetadata = function() {
          canvas.height = video.videoHeight
          canvas.width = video.videoWidth
          this.currentTime = Math.min(Math.max(0, (time < 0 ? this.duration : 0) + time), this.duration)
          resolve()
        }
      })
    }

    async function start(){
      start_time = new Date().getTime()
      window.start_time = start_time
      await load_video()
      worker.postMessage("https://unpkg.com/webm-wasm@0.4.1/dist/webm-wasm.wasm")
    }
    </script>
  </body>
</html>
</p>
      </div>
      <div class="content">
        <h2>truly distributed, local first apps</h2>
        <p>
        Datauri apps dont rely on http server for initialization. Many apps however load dynamic content from internet.
        Releasing datauri apps is very simple, because there is no need for web server or domain name.
        </p>
        <p>Using nostr as backbone for dynamic content provides distributed alternative for centralized solutions.
        It is simple to load content from nostr notes. Nostr is based on websocket connection. You can publish ~50 kB data on
        one nostr note, which is just json wrapper for any data content.
        See <a href="https://www.e2encrypted.com/nostr/nips/">nostr documentation</a>
        for more information.</p>

        <h2>immutable or updatable content</h2>
        <p>
        Datauris are by default immutable, ie. developer cannot update its content after its published,
        however this is not true for dynamically loaded content. Developer can use dynamic content loader
        to load full page source from nostr event. Below is an example.
        </p>
        <ul>
          <li>r = relay_uri</li>
          <li>k = pubkey</li>
          <li>f = d_tag</li>
        </ul>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=filepublish.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>
        <p>In this case, developer can publish new nostr event using same pubkey and same d tag value, which will override the previous content</p>
      </div>
    </main>
    <script>
    document.onclick = function(e){
      if(!e.target.classList.contains("data")){
        for(let datauri of document.querySelectorAll("p.data")){
          datauri.classList.remove("scroll")
        }
      }
    }

    for(let datauri of document.querySelectorAll("p.data")){
      datauri.onclick = function(){
        const range = document.createRange()
        range.selectNode(this)
        getSelection().removeAllRanges()
        getSelection().addRange(range)
        this.classList.add("scroll")
      }
    }

    function apply_hash(){
      const active_index = location.hash === "#devs" ? 1 : 0
      document.querySelector(".menu.active").classList.remove("active")
      document.querySelector(".content.active").classList.remove("active")
      document.querySelectorAll(".menu")[active_index].classList.add("active")
      document.querySelectorAll(".content")[active_index].classList.add("active")
    }

    addEventListener("hashchange", apply_hash)
    apply_hash()
    </script>
  </body>
</html>
#

some datauri #apps, as well as info for #devs

learn why #datauris are truly #distributed, #standalone #localfirst apps

nostr:note1mumewka2v8vkmyx3dts0z67qc4ws3s6jheywa7jjwhhzvrtyw8lqdzqvfq

Replying to Avatar b

#datauri_info #datauri #info

data:text/html;base64,<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <title>datauri info</title>
    <style>
      body {
        margin: 1em 1em 3em 1em;
      }

      header h1,
      header h2 {
        display: inline-block;
        margin-right: 1em;
      }

      .menu,
      .content {
        display: none;
      }

      .active {
        display: block;
      }

      h2 > a {
        padding: .2em .3em;
      }

      a:visited {
        color: inherit;
      }

      a:hover {
        color: rgb(200, 0, 0);
      }

      .data {
        background: rgb(220, 220, 100);
        /*background: rgb(180, 200, 230);*/
        /*background: rgb(180, 200, 230);*/
        overflow: hidden;
        word-break: break-all;
        max-height: 10em;
        padding: .3em;
      }

      .data,
      a {
        border-radius: .3em;
      }

      .data.scroll {
        overflow: scroll;
      }

      h2 {
        margin: 1em 0 .4em 0;
      }

      h2:first-child {
        margin-top: 0;
      }

      p {
        margin: 0 0 1em 0;
      }
    </style>
  </head>
  <body>
    <header>
      <div class="menu active">
        <h1>datauri info</h1>
        <h2>
          <a href="#devs">datauri for devs</a>
        </h2>
      </div>
      <div class="menu">
        <h1>datauri for devs</h1>
        <h2>
          <a href="#">datauri info</a>
        </h2>
      </div>
    </header>
    <main>
      <div class="content active">
        <p>
        What is datauri application? Apps hosted as datauris are basically portable web apps that work without http server.
        If you see this page, you probably already know how datauri apps work. You just copy-paste the link and access the app.
        Here are some datauri apps.
        </p>
        <h2>wired</h2>
        <p>Anon approved pow based nostr client</p>
        <p class="data">data:text/html;base64,PCFkb2N0eXBlIGh0bWw+PGh0bWwgbGFuZz0iZW4iPjxoZWFkPjxtZXRhIGNoYXJzZXQ9InV0Zi04Ii8+PG1ldGEgbmFtZT0idmlld3BvcnQiIGNvbnRlbnQ9IndpZHRoPWRldmljZS13aWR0aCxpbml0aWFsLXNjYWxlPTEiLz48bWV0YSBuYW1lPSJ0aGVtZS1jb2xvciIgY29udGVudD0iIzAwMDAwMCIvPjxtZXRhIG5hbWU9Im1vYmlsZS13ZWItYXBwLWNhcGFibGUiIGNvbnRlbnQ9InllcyIvPjxtZXRhIG5hbWU9ImFwcGxlLXRvdWNoLWZ1bGxzY3JlZW4iIGNvbnRlbnQ9InllcyIvPjxtZXRhIG5hbWU9ImFwcGxlLW1vYmlsZS13ZWItYXBwLXRpdGxlIiBjb250ZW50PSJFeHBvIi8+PG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+PG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtc3RhdHVzLWJhci1zdHlsZSIgY29udGVudD0iZGVmYXVsdCIvPjxtZXRhIG5hbWU9ImRlc2NyaXB0aW9uIiBjb250ZW50PSJUaGUgV2lyZWQiLz48dGl0bGU+VGhlIFdpcmVkPC90aXRsZT48L2hlYWQ+PGJvZHk+PG5vc2NyaXB0PllvdSBuZWVkIHRvIGVuYWJsZSBKYXZhU2NyaXB0IHRvIHJ1biB0aGlzIGFwcC48L25vc2NyaXB0PjxkaXYgaWQ9InJvb3QiPjwvZGl2PjxzY3JpcHQ+Cihhc3luYyBmdW5jdGlvbigpewogIGFzeW5jIGZ1bmN0aW9uIGRnZXQoaWQsIHJlbGF5KXsKICAgIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSkgPT4gewogICAgICBjb25zdCBzb2NrZXQgPSBuZXcgV2ViU29ja2V0KHJlbGF5KQogICAgICBsZXQgZmlsZV9kYXRhID0gW10KICAgICAgbGV0IGZpbGVpZHMgPSBbXQoKICAgICAgc29ja2V0Lm9ub3BlbiA9IGZ1bmN0aW9uKCl7CiAgICAgICAgc29ja2V0LnNlbmQoSlNPTi5zdHJpbmdpZnkoWyJSRVEiLCAiZmlsZWlkcyIsIHsiaWRzIjogW2lkXX1dKSkKICAgICAgfQoKICAgICAgc29ja2V0Lm9ubWVzc2FnZSA9IGFzeW5jIGZ1bmN0aW9uKGUpewogICAgICAgIGNvbnN0IGRhdGEgPSBKU09OLnBhcnNlKGUuZGF0YSkKCiAgICAgICAgaWYoZGF0YVswXSA9PT0gIkVWRU5UIiAmJiBkYXRhWzFdID09PSAiZmlsZWlkcyIpewogICAgICAgICAgZmlsZWlkcy5wdXNoKC4uLkpTT04ucGFyc2UoZGF0YVsyXS5jb250ZW50KS5maWxlaWRzKQogICAgICAgICAgc29ja2V0LnNlbmQoSlNPTi5zdHJpbmdpZnkoWyJSRVEiLCAicGFydHMiLCB7ImlkcyI6IGZpbGVpZHMsICJsaW1pdCI6IDMwMDB9XSkpCiAgICAgICAgfQogICAgICAgIGVsc2UgaWYoZGF0YVswXSA9PT0gIkVWRU5UIiAmJiBkYXRhWzFdID09PSAicGFydHMiKXsKICAgICAgICAgIGNvbnN0IG1hdGNoID0gZGF0YVsyXS5jb250ZW50Lm1hdGNoKC9ecGFydChcZHsxLDR9KTsoXGR7MSw0fSk7W2EtejAtOV0rOy8pCgogICAgICAgICAgaWYobWF0Y2gpewogICAgICAgICAgICBjb25zdCBwYXJ0X2RhdGEgPSBkYXRhWzJdLmNvbnRlbnQuc3Vic3RyKG1hdGNoWzBdLmxlbmd0aCkKICAgICAgICAgICAgZmlsZV9kYXRhLnB1c2goW3BhcnNlSW50KG1hdGNoWzFdLCAxMCksIHBhcnRfZGF0YV0pCiAgICAgICAgICB9ZWxzZXsKICAgICAgICAgICAgZmlsZV9kYXRhLnB1c2goWzEsIGRhdGFbMl0uY29udGVudF0pCiAgICAgICAgICB9CgogICAgICAgICAgaWYoZmlsZV9kYXRhLmxlbmd0aCA9PT0gZmlsZWlkcy5sZW5ndGgpewogICAgICAgICAgICBmaWxlX2RhdGEgPSBmaWxlX2RhdGEuc29ydCgoYSwgYikgPT4gYVswXSAtIGJbMF0pLm1hcChlID0+IGVbMV0pCiAgICAgICAgICAgIGNvbnN0IGRhdGF1cmkgPSAiZGF0YToiICsgZmlsZV9kYXRhLmpvaW4oIiIpCiAgICAgICAgICAgIGNvbnN0IGJsb2IgPSBhd2FpdCAoYXdhaXQgZmV0Y2goZGF0YXVyaSkpLmJsb2IoKQogICAgICAgICAgICByZXNvbHZlKGJsb2IpCiAgICAgICAgICB9CiAgICAgICAgfQogICAgICB9CiAgICB9KQogIH0KCiAgY29uc3Qgd29ya2VyX3NvdXJjZSA9IGF3YWl0IGRnZXQoImZlZWNlYTAxZmU2ZDkyMGU5ZjU2M2U5OWU1MjdhMmE1MjVlODQwOTUyN2ViMjc3Yzk0ZDJlOTA3MzRkYzRjNTAiLCAid3NzOi8vbm9zLmxvbCIpCiAgY29uc3Qgd29ya2VyXzU5Ml91cmwgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGF3YWl0IGRnZXQoImZlMGVhNjA2ODBiMTYwYjYyN2M0YmYzNThlMDk1YWZhMTcxNWJhMTlhNWI5YzkwOTYzMGQ1MmQzZTQzMjQ3MmMiLCAid3NzOi8vbm9zLmxvbCIpKQogIGNvbnN0IGljb25fcG5nX3VybCA9IFVSTC5jcmVhdGVPYmplY3RVUkwoYXdhaXQgZGdldCgiMDVkZTM0ZWE5MDkyMmVlOTQxMDUzYzQzYjUwMGRhYTE0ODRlMWY3MzM2M2ZjNWRhNzIyMzRiNDAzYjk3NzM3OCIsICJ3c3M6Ly9ub3MubG9sIikpCiAgY29uc3QgbWFpbl9qc19tYXBfdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChhd2FpdCBkZ2V0KCI1NWVkNDdiMjk0NzJkZTE4Y2I3MWQ0M2NjOGNmZmU3NTcwNzJmNDA3Y2ZlOGNlYjRkMDQyNDdiZGU0NDc1M2VlIiwgIndzczovL25vcy5sb2wiKSkKCiAgbGV0IHdvcmtlcl9zb3VyY2Vfc3RyID0gYXdhaXQgd29ya2VyX3NvdXJjZS50ZXh0KCkKICB3b3JrZXJfc291cmNlX3N0ciA9IHdvcmtlcl9zb3VyY2Vfc3RyLnJlcGxhY2UoJ2ltcG9ydFNjcmlwdHModC5wK3QudShyKSknLCAnaW1wb3J0U2NyaXB0cygiJyArIHdvcmtlcl81OTJfdXJsICsgJyIpJykKICBjb25zdCB3b3JrZXJfdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChuZXcgQmxvYihbd29ya2VyX3NvdXJjZV9zdHJdLCB7dHlwZTogInRleHQvamF2YXNjcmlwdCJ9KSkKCiAgY29uc3Qgb3JpZ2luYWxXb3JrZXJDb25zdHJ1Y3RvciA9IFdvcmtlcjsKCiAgV29ya2VyID0gZnVuY3Rpb24odXJsKSB7CiAgICB1cmwgPSB3b3JrZXJfdXJsCiAgICByZXR1cm4gbmV3IG9yaWdpbmFsV29ya2VyQ29uc3RydWN0b3IodXJsKTsKICB9CgogIFVSTCA9IGNsYXNzIE15VVJMIGV4dGVuZHMgVVJMIHsKICAgIGNvbnN0cnVjdG9yKGlucHV0KSB7CiAgICAgIHN1cGVyKGlucHV0LmluZGV4T2YoIi8iKSA9PT0gMCAmJiAobG9jYXRpb24ucHJvdG9jb2wgKyBsb2NhdGlvbi5wYXRobmFtZSArIGlucHV0KSB8fCBpbnB1dCk7CiAgICB9CiAgfQoKICBkYXRhTG9jYWxTdG9yYWdlID0gewogICAgZ2V0SXRlbTogZnVuY3Rpb24oKXt9LAogICAgc2V0SXRlbTogZnVuY3Rpb24oKXt9CiAgfQoKICBjb25zdCBzY3JpcHRzID0gWwogICAgIjE1NDNhMDcwYTQyZThlMGMwNTg2MTg0MGJkZTdlNWI2OTFjNGI1ZGVjNjYyNzQ1ODUyYzIyODNjZmVmZDhjYTUiCiAgXQoKICBjb25zdCBzdHlsZXMgPSBbCiAgICAiNjk5NjA3NzRjZTZkZmE4MzFhMzMwODczNDEyNWQ4MGRlMjY5YTg0ZjU5ODg4YTdkMzEyMTVlYTFhYzU1MTE1MiIKICBdCgogIGNvbnN0IHNjcmlwdF9pZCA9IHNjcmlwdHNbMF0KICBjb25zdCBzY3JpcHRfc291cmNlID0gYXdhaXQgZGdldChzY3JpcHRfaWQsICJ3c3M6Ly9ub3MubG9sIikKICBsZXQgc2NyaXB0X3NvdXJjZV9zdHIgPSBhd2FpdCBzY3JpcHRfc291cmNlLnRleHQoKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZSgiL2ljb24ucG5nIiwgaWNvbl9wbmdfdXJsKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZSgvKHNvdXJjZU1hcHBpbmdVUkw9KShtYWluXC5bYS16MC05XXs4fVwuanNcLm1hcCkvLCAiJDFodHRwOi8vbG9jYWxob3N0OjgwMDAvJDIiKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZUFsbCgibG9jYWxTdG9yYWdlIiwgImRhdGFMb2NhbFN0b3JhZ2UiKQoKICBjb25zdCBzY3JpcHRfZWwgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCJzY3JpcHQiKQogIHNjcmlwdF9lbC5zcmMgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKG5ldyBCbG9iKFtzY3JpcHRfc291cmNlX3N0cl0sIHt0eXBlOiAidGV4dC9qYXZhc2NyaXB0In0pKQogIGNvbnNvbGUubG9nKCJzY3JpcHRfZWwuc3JjIiwgc2NyaXB0X2VsLnNyYykKICBkb2N1bWVudC5ib2R5LmFwcGVuZChzY3JpcHRfZWwpCgogIGZvcihsZXQgc3R5bGVfaWQgb2Ygc3R5bGVzKXsKICAgIGNvbnN0IHN0eWxlX3NvdXJjZSA9IGF3YWl0IGRnZXQoc3R5bGVfaWQsICJ3c3M6Ly9ub3MubG9sIikKICAgIGNvbnN0IHN0eWxlX2VsID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgibGluayIpCiAgICBzdHlsZV9lbC5yZWwgPSAic3R5bGVzaGVldCIKICAgIHN0eWxlX2VsLmhyZWYgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGF3YWl0IHN0eWxlX3NvdXJjZSkKICAgIGRvY3VtZW50LmJvZHkuYXBwZW5kKHN0eWxlX2VsKQogIH0KfSkoKQo8L3NjcmlwdD4KPC9ib2R5PjwvaHRtbD4K</p>

        <h2>ourchan</h2>
        <p>Anon approved image board</p>
        <p class="data">data:text/html;r=nos.lol;k=2b8d50fcc974ffb652c0ab33edebb6a9ce72cd154591969592bc8b607a4ea1f2;f=ourchan;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>filepublish</h2>
        <p>Upload files</p>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=filepublish.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>fileloader</h2>
        <p>Download/view files</p>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=fileloader.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>video clip creator</h2>
        <p>Create and upload 1 second video clips</p>
        <p class="data">data:text/html;base64,<!DOCTYPE html>
<html>
  <head>
    <title>video clip creator</title>
    <style>
    body {
      margin: 1em;
    }
    h1 {
      margin-top: 0;
      font-size: 1.5em;
    }
    #log {
      overflow: auto;
      position: absolute;
      bottom: 1em;
      top: 11em;
      border: .3em solid #666;
      border-radius: .3em;
      right: 1em;
      left: 1em;
      padding: .5em;
      background: #eee;
    }
    #progress {
      background: #ddd;
      border-radius: .3em;
    }
    #progressbar {
      height: 1.3em;
      background: blue;
      width: 0%;
      border-radius: .4em;
    }
    #log > a {
      margin-bottom: 1em;
      display: block;
    }
    p {
      margin-top: 0;
      max-height: 7em;
      overflow: scroll;
    }
    video {
      display: none;
      z-index: 1;
    }
    #canvas,
    video {
      position: absolute;
      top: 1em;
      right: 1em;
      width: 10em;
      height: 6em;
    }
    </style>
  </head>
  <body>
    <h1>video clip creator</h1>
    <form id="form">
      <p>
        <input id="file" type="file"/>
      </p>
      <p>
        <input id="submit" type="submit" value="encode"/>
      </p>
    </form>
    <video id="video"></video>
    <video id="video2" controls loop="1"></video>
    <canvas id="canvas"></canvas>
    <div id="progress">
      <div id="progressbar"></div>
    </div>
    <div id="log"><div>
    <script type="module">
    const canvas = document.getElementById("canvas")
    const ctx = canvas.getContext('2d')
    const log = document.getElementById("log")
    const video = document.getElementById("video")
    const video2 = document.getElementById("video2")
    const progressbar = document.getElementById("progressbar")
    let files = null

    const frame_duration = 1/30
    const extract_frame_count = 30//1 / frame_duration
    let time = 15

    const buffer = await fetch(
      "https://unpkg.com/webm-wasm@0.4.1/dist/webm-worker.js"
    ).then(r => r.arrayBuffer())

    const worker = new Worker(
      URL.createObjectURL(new Blob([buffer], { type: "text/javascript" }))
    )

    let encoded_data = []
    let start_time = null
    let ready = false

    document.getElementById("form").onsubmit = function(e){
      e.preventDefault()
      video2.style.display = "none";
      video2.pause()
      progressbar.style.width = "0.5%"
      this.querySelectorAll("input").forEach(el => el.disabled = true)
      document.getElementById("submit").value = "encoding"
      files = document.getElementById("file").files
      start()
    }

    function blobToDataURL(blob, callback) {
      return new Promise(resolve => {
        var a = new FileReader()
        a.onload = function(e) {
          resolve(e.target.result)
        }
        a.readAsDataURL(blob)
      })
    }

    function info(msg, type){
      const el = document.createElement(type === 2 && "a" || "p")
      el[(type === 3) && "innerHTML" || "innerText"] = msg

      if(type){
        el.href = msg
      }

      log.append(el)
      log.scrollTo(0, 1e6)
    }

    async function process_video(){
      let frame_count = 0

      video.onseeked = function(e) {
        frame_count++
        info("extract frame " + frame_count + " / " + extract_frame_count)

        ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
        const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height)

        if(frame_count < extract_frame_count){
          time += frame_duration
          this.currentTime = Math.min(Math.max(0, (time < 0 ? this.duration : 0) + time), this.duration)
          worker.postMessage(imageData.data.buffer, [imageData.data.buffer])
        }else{
          worker.postMessage(null)
        }
      }
    }

    function total_time(start){
      return (new Date().getTime() - start) / 1000
    }

    function createBufferURL(buffer, type = '') {
      return URL.createObjectURL(new Blob([buffer], {type}));
    }

    async function show_data_uri(blob){
      let blob_url = URL.createObjectURL(blob)

      info(await blobToDataURL(blob))
      info(blob_url, 2)

      video2.src = blob_url
      video2.style.display = "block";
      video2.play()

      progressbar.style.width = "100%"
      document.getElementById("form").reset()
      document.getElementById("submit").value = "encode"
      document.querySelectorAll("form input").forEach(el => el.disabled = false)
    }

    worker.onmessage = function(e){
      if(!e.data) {
        return
      }

      if(e.data != null && typeof e.data == "object" && e.data.byteLength > 0){
        encoded_data.push(e.data)
        info("encode " + encoded_data.length + " / " + extract_frame_count)
        progressbar.style.width = parseInt((encoded_data.length / extract_frame_count) * 100, 10) + "%"
      }

      if(e.data == "READY"){
        info("ready", canvas.width, canvas.height)

        worker.postMessage({
          width: canvas.width,
          height: canvas.height,
          realtime: true,
          bitrate: 100
        })

        //for(let i = 0; i < frames.length; i++){
        //  worker.postMessage(frames[i], [frames[i]])
        //}

        ready = true
        process_video()
      }
      else if(ready){
        if(e.data.byteLength == 0){
          info("encoding finished (duration: " + parseInt(total_time(start_time), 10) +
            " s, speed: " + parseInt(total_time(start_time) / extract_frame_count, 10) +" s / frame)")
          let blob = new Blob(encoded_data, { type: 'video/webm' })
          show_data_uri(blob)
        }
      }
    }

    function load_video(){
      return new Promise(resolve => {
        video.src = URL.createObjectURL(files[0])

        video.onloadedmetadata = function() {
          canvas.height = video.videoHeight
          canvas.width = video.videoWidth
          this.currentTime = Math.min(Math.max(0, (time < 0 ? this.duration : 0) + time), this.duration)
          resolve()
        }
      })
    }

    async function start(){
      start_time = new Date().getTime()
      window.start_time = start_time
      await load_video()
      worker.postMessage("https://unpkg.com/webm-wasm@0.4.1/dist/webm-wasm.wasm")
    }
    </script>
  </body>
</html>
</p>
      </div>
      <div class="content">
        <h2>truly distributed, local first apps</h2>
        <p>
        Datauri apps dont rely on http server for initialization. Many apps however load dynamic content from internet.
        Releasing datauri apps is very simple, because there is no need for web server or domain name.
        </p>
        <p>Using nostr as backbone for dynamic content provides distributed alternative for centralized solutions.
        It is simple to load content from nostr notes. Nostr is based on websocket connection. You can publish ~50 kB data on
        one nostr note, which is just json wrapper for any data content.
        See <a href="https://www.e2encrypted.com/nostr/nips/">nostr documentation</a>
        for more information.</p>

        <h2>immutable or updatable content</h2>
        <p>
        Datauris are by default immutable, ie. developer cannot update its content after its published,
        however this is not true for dynamically loaded content. Developer can use dynamic content loader
        to load full page source from nostr event. Below is an example.
        </p>
        <ul>
          <li>r = relay_uri</li>
          <li>k = pubkey</li>
          <li>f = d_tag</li>
        </ul>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=filepublish.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>
        <p>In this case, developer can publish new nostr event using same pubkey and same d tag value, which will override the previous content</p>
      </div>
    </main>
    <script>
    document.onclick = function(e){
      if(!e.target.classList.contains("data")){
        for(let datauri of document.querySelectorAll("p.data")){
          datauri.classList.remove("scroll")
        }
      }
    }

    for(let datauri of document.querySelectorAll("p.data")){
      datauri.onclick = function(){
        const range = document.createRange()
        range.selectNode(this)
        getSelection().removeAllRanges()
        getSelection().addRange(range)
        this.classList.add("scroll")
      }
    }

    function apply_hash(){
      const active_index = location.hash === "#devs" ? 1 : 0
      document.querySelector(".menu.active").classList.remove("active")
      document.querySelector(".content.active").classList.remove("active")
      document.querySelectorAll(".menu")[active_index].classList.add("active")
      document.querySelectorAll(".content")[active_index].classList.add("active")
    }

    addEventListener("hashchange", apply_hash)
    apply_hash()
    </script>
  </body>
</html>
#

data:text/html;base64,<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <title>datauri info</title>
    <style>
      body {
        margin: 1em 1em 3em 1em;
      }

      header h1,
      header h2 {
        display: inline-block;
        margin-right: 1em;
      }

      .menu,
      .content {
        display: none;
      }

      .active {
        display: block;
      }

      h2 > a {
        padding: .2em .3em;
      }

      a, a:visited {
        color: inherit;
      }

      a:hover {
        color: rgb(200, 0, 0);
      }

      .data {
        background: rgb(220, 220, 100);
        /*background: rgb(180, 200, 230);*/
        /*background: rgb(180, 200, 230);*/
        overflow: hidden;
        word-break: break-all;
        max-height: 10em;
        padding: .3em;
      }

      .data,
      a {
        border-radius: .3em;
      }

      .data.scroll {
        overflow: scroll;
      }

      h2 {
        margin: 1em 0 .4em 0;
      }

      h2:first-child {
        margin-top: 0;
      }

      p {
        margin: 0 0 1em 0;
      }
    </style>
  </head>
  <body>
    <header>
      <div class="menu active">
        <h1>datauri info</h1>
        <h2>
          <a href="#devs">datauri for devs</a>
        </h2>
      </div>
      <div class="menu">
        <h1>datauri for devs</h1>
        <h2>
          <a href="#">datauri info</a>
        </h2>
      </div>
    </header>
    <main>
      <div class="content active">
        <p>
        What is datauri application? Apps hosted as datauris are basically portable web apps that work without http server.
        If you see this page, you probably already know how datauri apps work. You just copy-paste the link and access the app.
        Here are some datauri apps.
        </p>
        <h2>wired</h2>
        <p>Anon approved pow based nostr client</p>
        <p class="data">data:text/html;base64,PCFkb2N0eXBlIGh0bWw+PGh0bWwgbGFuZz0iZW4iPjxoZWFkPjxtZXRhIGNoYXJzZXQ9InV0Zi04Ii8+PG1ldGEgbmFtZT0idmlld3BvcnQiIGNvbnRlbnQ9IndpZHRoPWRldmljZS13aWR0aCxpbml0aWFsLXNjYWxlPTEiLz48bWV0YSBuYW1lPSJ0aGVtZS1jb2xvciIgY29udGVudD0iIzAwMDAwMCIvPjxtZXRhIG5hbWU9Im1vYmlsZS13ZWItYXBwLWNhcGFibGUiIGNvbnRlbnQ9InllcyIvPjxtZXRhIG5hbWU9ImFwcGxlLXRvdWNoLWZ1bGxzY3JlZW4iIGNvbnRlbnQ9InllcyIvPjxtZXRhIG5hbWU9ImFwcGxlLW1vYmlsZS13ZWItYXBwLXRpdGxlIiBjb250ZW50PSJFeHBvIi8+PG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+PG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtc3RhdHVzLWJhci1zdHlsZSIgY29udGVudD0iZGVmYXVsdCIvPjxtZXRhIG5hbWU9ImRlc2NyaXB0aW9uIiBjb250ZW50PSJUaGUgV2lyZWQiLz48dGl0bGU+VGhlIFdpcmVkPC90aXRsZT48L2hlYWQ+PGJvZHk+PG5vc2NyaXB0PllvdSBuZWVkIHRvIGVuYWJsZSBKYXZhU2NyaXB0IHRvIHJ1biB0aGlzIGFwcC48L25vc2NyaXB0PjxkaXYgaWQ9InJvb3QiPjwvZGl2PjxzY3JpcHQ+Cihhc3luYyBmdW5jdGlvbigpewogIGFzeW5jIGZ1bmN0aW9uIGRnZXQoaWQsIHJlbGF5KXsKICAgIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSkgPT4gewogICAgICBjb25zdCBzb2NrZXQgPSBuZXcgV2ViU29ja2V0KHJlbGF5KQogICAgICBsZXQgZmlsZV9kYXRhID0gW10KICAgICAgbGV0IGZpbGVpZHMgPSBbXQoKICAgICAgc29ja2V0Lm9ub3BlbiA9IGZ1bmN0aW9uKCl7CiAgICAgICAgc29ja2V0LnNlbmQoSlNPTi5zdHJpbmdpZnkoWyJSRVEiLCAiZmlsZWlkcyIsIHsiaWRzIjogW2lkXX1dKSkKICAgICAgfQoKICAgICAgc29ja2V0Lm9ubWVzc2FnZSA9IGFzeW5jIGZ1bmN0aW9uKGUpewogICAgICAgIGNvbnN0IGRhdGEgPSBKU09OLnBhcnNlKGUuZGF0YSkKCiAgICAgICAgaWYoZGF0YVswXSA9PT0gIkVWRU5UIiAmJiBkYXRhWzFdID09PSAiZmlsZWlkcyIpewogICAgICAgICAgZmlsZWlkcy5wdXNoKC4uLkpTT04ucGFyc2UoZGF0YVsyXS5jb250ZW50KS5maWxlaWRzKQogICAgICAgICAgc29ja2V0LnNlbmQoSlNPTi5zdHJpbmdpZnkoWyJSRVEiLCAicGFydHMiLCB7ImlkcyI6IGZpbGVpZHMsICJsaW1pdCI6IDMwMDB9XSkpCiAgICAgICAgfQogICAgICAgIGVsc2UgaWYoZGF0YVswXSA9PT0gIkVWRU5UIiAmJiBkYXRhWzFdID09PSAicGFydHMiKXsKICAgICAgICAgIGNvbnN0IG1hdGNoID0gZGF0YVsyXS5jb250ZW50Lm1hdGNoKC9ecGFydChcZHsxLDR9KTsoXGR7MSw0fSk7W2EtejAtOV0rOy8pCgogICAgICAgICAgaWYobWF0Y2gpewogICAgICAgICAgICBjb25zdCBwYXJ0X2RhdGEgPSBkYXRhWzJdLmNvbnRlbnQuc3Vic3RyKG1hdGNoWzBdLmxlbmd0aCkKICAgICAgICAgICAgZmlsZV9kYXRhLnB1c2goW3BhcnNlSW50KG1hdGNoWzFdLCAxMCksIHBhcnRfZGF0YV0pCiAgICAgICAgICB9ZWxzZXsKICAgICAgICAgICAgZmlsZV9kYXRhLnB1c2goWzEsIGRhdGFbMl0uY29udGVudF0pCiAgICAgICAgICB9CgogICAgICAgICAgaWYoZmlsZV9kYXRhLmxlbmd0aCA9PT0gZmlsZWlkcy5sZW5ndGgpewogICAgICAgICAgICBmaWxlX2RhdGEgPSBmaWxlX2RhdGEuc29ydCgoYSwgYikgPT4gYVswXSAtIGJbMF0pLm1hcChlID0+IGVbMV0pCiAgICAgICAgICAgIGNvbnN0IGRhdGF1cmkgPSAiZGF0YToiICsgZmlsZV9kYXRhLmpvaW4oIiIpCiAgICAgICAgICAgIGNvbnN0IGJsb2IgPSBhd2FpdCAoYXdhaXQgZmV0Y2goZGF0YXVyaSkpLmJsb2IoKQogICAgICAgICAgICByZXNvbHZlKGJsb2IpCiAgICAgICAgICB9CiAgICAgICAgfQogICAgICB9CiAgICB9KQogIH0KCiAgY29uc3Qgd29ya2VyX3NvdXJjZSA9IGF3YWl0IGRnZXQoImZlZWNlYTAxZmU2ZDkyMGU5ZjU2M2U5OWU1MjdhMmE1MjVlODQwOTUyN2ViMjc3Yzk0ZDJlOTA3MzRkYzRjNTAiLCAid3NzOi8vbm9zLmxvbCIpCiAgY29uc3Qgd29ya2VyXzU5Ml91cmwgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGF3YWl0IGRnZXQoImZlMGVhNjA2ODBiMTYwYjYyN2M0YmYzNThlMDk1YWZhMTcxNWJhMTlhNWI5YzkwOTYzMGQ1MmQzZTQzMjQ3MmMiLCAid3NzOi8vbm9zLmxvbCIpKQogIGNvbnN0IGljb25fcG5nX3VybCA9IFVSTC5jcmVhdGVPYmplY3RVUkwoYXdhaXQgZGdldCgiMDVkZTM0ZWE5MDkyMmVlOTQxMDUzYzQzYjUwMGRhYTE0ODRlMWY3MzM2M2ZjNWRhNzIyMzRiNDAzYjk3NzM3OCIsICJ3c3M6Ly9ub3MubG9sIikpCiAgY29uc3QgbWFpbl9qc19tYXBfdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChhd2FpdCBkZ2V0KCI1NWVkNDdiMjk0NzJkZTE4Y2I3MWQ0M2NjOGNmZmU3NTcwNzJmNDA3Y2ZlOGNlYjRkMDQyNDdiZGU0NDc1M2VlIiwgIndzczovL25vcy5sb2wiKSkKCiAgbGV0IHdvcmtlcl9zb3VyY2Vfc3RyID0gYXdhaXQgd29ya2VyX3NvdXJjZS50ZXh0KCkKICB3b3JrZXJfc291cmNlX3N0ciA9IHdvcmtlcl9zb3VyY2Vfc3RyLnJlcGxhY2UoJ2ltcG9ydFNjcmlwdHModC5wK3QudShyKSknLCAnaW1wb3J0U2NyaXB0cygiJyArIHdvcmtlcl81OTJfdXJsICsgJyIpJykKICBjb25zdCB3b3JrZXJfdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChuZXcgQmxvYihbd29ya2VyX3NvdXJjZV9zdHJdLCB7dHlwZTogInRleHQvamF2YXNjcmlwdCJ9KSkKCiAgY29uc3Qgb3JpZ2luYWxXb3JrZXJDb25zdHJ1Y3RvciA9IFdvcmtlcjsKCiAgV29ya2VyID0gZnVuY3Rpb24odXJsKSB7CiAgICB1cmwgPSB3b3JrZXJfdXJsCiAgICByZXR1cm4gbmV3IG9yaWdpbmFsV29ya2VyQ29uc3RydWN0b3IodXJsKTsKICB9CgogIFVSTCA9IGNsYXNzIE15VVJMIGV4dGVuZHMgVVJMIHsKICAgIGNvbnN0cnVjdG9yKGlucHV0KSB7CiAgICAgIHN1cGVyKGlucHV0LmluZGV4T2YoIi8iKSA9PT0gMCAmJiAobG9jYXRpb24ucHJvdG9jb2wgKyBsb2NhdGlvbi5wYXRobmFtZSArIGlucHV0KSB8fCBpbnB1dCk7CiAgICB9CiAgfQoKICBkYXRhTG9jYWxTdG9yYWdlID0gewogICAgZ2V0SXRlbTogZnVuY3Rpb24oKXt9LAogICAgc2V0SXRlbTogZnVuY3Rpb24oKXt9CiAgfQoKICBjb25zdCBzY3JpcHRzID0gWwogICAgIjE1NDNhMDcwYTQyZThlMGMwNTg2MTg0MGJkZTdlNWI2OTFjNGI1ZGVjNjYyNzQ1ODUyYzIyODNjZmVmZDhjYTUiCiAgXQoKICBjb25zdCBzdHlsZXMgPSBbCiAgICAiNjk5NjA3NzRjZTZkZmE4MzFhMzMwODczNDEyNWQ4MGRlMjY5YTg0ZjU5ODg4YTdkMzEyMTVlYTFhYzU1MTE1MiIKICBdCgogIGNvbnN0IHNjcmlwdF9pZCA9IHNjcmlwdHNbMF0KICBjb25zdCBzY3JpcHRfc291cmNlID0gYXdhaXQgZGdldChzY3JpcHRfaWQsICJ3c3M6Ly9ub3MubG9sIikKICBsZXQgc2NyaXB0X3NvdXJjZV9zdHIgPSBhd2FpdCBzY3JpcHRfc291cmNlLnRleHQoKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZSgiL2ljb24ucG5nIiwgaWNvbl9wbmdfdXJsKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZSgvKHNvdXJjZU1hcHBpbmdVUkw9KShtYWluXC5bYS16MC05XXs4fVwuanNcLm1hcCkvLCAiJDFodHRwOi8vbG9jYWxob3N0OjgwMDAvJDIiKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZUFsbCgibG9jYWxTdG9yYWdlIiwgImRhdGFMb2NhbFN0b3JhZ2UiKQoKICBjb25zdCBzY3JpcHRfZWwgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCJzY3JpcHQiKQogIHNjcmlwdF9lbC5zcmMgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKG5ldyBCbG9iKFtzY3JpcHRfc291cmNlX3N0cl0sIHt0eXBlOiAidGV4dC9qYXZhc2NyaXB0In0pKQogIGNvbnNvbGUubG9nKCJzY3JpcHRfZWwuc3JjIiwgc2NyaXB0X2VsLnNyYykKICBkb2N1bWVudC5ib2R5LmFwcGVuZChzY3JpcHRfZWwpCgogIGZvcihsZXQgc3R5bGVfaWQgb2Ygc3R5bGVzKXsKICAgIGNvbnN0IHN0eWxlX3NvdXJjZSA9IGF3YWl0IGRnZXQoc3R5bGVfaWQsICJ3c3M6Ly9ub3MubG9sIikKICAgIGNvbnN0IHN0eWxlX2VsID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgibGluayIpCiAgICBzdHlsZV9lbC5yZWwgPSAic3R5bGVzaGVldCIKICAgIHN0eWxlX2VsLmhyZWYgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGF3YWl0IHN0eWxlX3NvdXJjZSkKICAgIGRvY3VtZW50LmJvZHkuYXBwZW5kKHN0eWxlX2VsKQogIH0KfSkoKQo8L3NjcmlwdD4KPC9ib2R5PjwvaHRtbD4K</p>

        <h2>ourchan</h2>
        <p>Anon approved image board</p>
        <p class="data">data:text/html;r=nos.lol;k=2b8d50fcc974ffb652c0ab33edebb6a9ce72cd154591969592bc8b607a4ea1f2;f=ourchan;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>filepublish</h2>
        <p>Upload files</p>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=filepublish.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>fileloader</h2>
        <p>Download/view files</p>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=fileloader.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>video clip creator</h2>
        <p>Create and upload 1 second video clips</p>
        <p class="data">data:text/html;base64,<!DOCTYPE html>
<html>
  <head>
    <title>video clip creator</title>
    <style>
    body {
      margin: 1em;
    }
    h1 {
      margin-top: 0;
      font-size: 1.5em;
    }
    #log {
      overflow: auto;
      position: absolute;
      bottom: 1em;
      top: 11em;
      border: .3em solid #666;
      border-radius: .3em;
      right: 1em;
      left: 1em;
      padding: .5em;
      background: #eee;
    }
    #progress {
      background: #ddd;
      border-radius: .3em;
    }
    #progressbar {
      height: 1.3em;
      background: blue;
      width: 0%;
      border-radius: .4em;
    }
    #log > a {
      margin-bottom: 1em;
      display: block;
    }
    p {
      margin-top: 0;
      max-height: 7em;
      overflow: scroll;
    }
    video {
      display: none;
      z-index: 1;
    }
    #canvas,
    video {
      position: absolute;
      top: 1em;
      right: 1em;
      width: 10em;
      height: 6em;
    }
    </style>
  </head>
  <body>
    <h1>video clip creator</h1>
    <form id="form">
      <p>
        <input id="file" type="file"/>
      </p>
      <p>
        <input id="submit" type="submit" value="encode"/>
      </p>
    </form>
    <video id="video"></video>
    <video id="video2" controls loop="1"></video>
    <canvas id="canvas"></canvas>
    <div id="progress">
      <div id="progressbar"></div>
    </div>
    <div id="log"><div>
    <script type="module">
    const canvas = document.getElementById("canvas")
    const ctx = canvas.getContext('2d')
    const log = document.getElementById("log")
    const video = document.getElementById("video")
    const video2 = document.getElementById("video2")
    const progressbar = document.getElementById("progressbar")
    let files = null

    const frame_duration = 1/30
    const extract_frame_count = 30//1 / frame_duration
    let time = 15

    const buffer = await fetch(
      "https://unpkg.com/webm-wasm@0.4.1/dist/webm-worker.js"
    ).then(r => r.arrayBuffer())

    const worker = new Worker(
      URL.createObjectURL(new Blob([buffer], { type: "text/javascript" }))
    )

    let encoded_data = []
    let start_time = null
    let ready = false

    document.getElementById("form").onsubmit = function(e){
      e.preventDefault()
      video2.style.display = "none";
      video2.pause()
      progressbar.style.width = "0.5%"
      this.querySelectorAll("input").forEach(el => el.disabled = true)
      document.getElementById("submit").value = "encoding"
      files = document.getElementById("file").files
      start()
    }

    function blobToDataURL(blob, callback) {
      return new Promise(resolve => {
        var a = new FileReader()
        a.onload = function(e) {
          resolve(e.target.result)
        }
        a.readAsDataURL(blob)
      })
    }

    function info(msg, type){
      const el = document.createElement(type === 2 && "a" || "p")
      el[(type === 3) && "innerHTML" || "innerText"] = msg

      if(type){
        el.href = msg
      }

      log.append(el)
      log.scrollTo(0, 1e6)
    }

    async function process_video(){
      let frame_count = 0

      video.onseeked = function(e) {
        frame_count++
        info("extract frame " + frame_count + " / " + extract_frame_count)

        ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
        const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height)

        if(frame_count < extract_frame_count){
          time += frame_duration
          this.currentTime = Math.min(Math.max(0, (time < 0 ? this.duration : 0) + time), this.duration)
          worker.postMessage(imageData.data.buffer, [imageData.data.buffer])
        }else{
          worker.postMessage(null)
        }
      }
    }

    function total_time(start){
      return (new Date().getTime() - start) / 1000
    }

    function createBufferURL(buffer, type = '') {
      return URL.createObjectURL(new Blob([buffer], {type}));
    }

    async function show_data_uri(blob){
      let blob_url = URL.createObjectURL(blob)

      info(await blobToDataURL(blob))
      info(blob_url, 2)

      video2.src = blob_url
      video2.style.display = "block";
      video2.play()

      progressbar.style.width = "100%"
      document.getElementById("form").reset()
      document.getElementById("submit").value = "encode"
      document.querySelectorAll("form input").forEach(el => el.disabled = false)
    }

    worker.onmessage = function(e){
      if(!e.data) {
        return
      }

      if(e.data != null && typeof e.data == "object" && e.data.byteLength > 0){
        encoded_data.push(e.data)
        info("encode " + encoded_data.length + " / " + extract_frame_count)
        progressbar.style.width = parseInt((encoded_data.length / extract_frame_count) * 100, 10) + "%"
      }

      if(e.data == "READY"){
        info("ready", canvas.width, canvas.height)

        worker.postMessage({
          width: canvas.width,
          height: canvas.height,
          realtime: true,
          bitrate: 100
        })

        //for(let i = 0; i < frames.length; i++){
        //  worker.postMessage(frames[i], [frames[i]])
        //}

        ready = true
        process_video()
      }
      else if(ready){
        if(e.data.byteLength == 0){
          info("encoding finished (duration: " + parseInt(total_time(start_time), 10) +
            " s, speed: " + parseInt(total_time(start_time) / extract_frame_count, 10) +" s / frame)")
          let blob = new Blob(encoded_data, { type: 'video/webm' })
          show_data_uri(blob)
        }
      }
    }

    function load_video(){
      return new Promise(resolve => {
        video.src = URL.createObjectURL(files[0])

        video.onloadedmetadata = function() {
          canvas.height = video.videoHeight
          canvas.width = video.videoWidth
          this.currentTime = Math.min(Math.max(0, (time < 0 ? this.duration : 0) + time), this.duration)
          resolve()
        }
      })
    }

    async function start(){
      start_time = new Date().getTime()
      window.start_time = start_time
      await load_video()
      worker.postMessage("https://unpkg.com/webm-wasm@0.4.1/dist/webm-wasm.wasm")
    }
    </script>
  </body>
</html>
</p>
      </div>
      <div class="content">
        <h2>truly distributed, local first apps</h2>
        <p>
        Datauri apps dont rely on http server for initialization. Many apps however load dynamic content from internet.
        Releasing datauri apps is very simple, because there is no need for web server or domain name.
        </p>
        <p>Using nostr as backbone for dynamic content provides distributed alternative for centralized solutions.
        It is simple to load content from nostr notes. Nostr is based on websocket connection. You can publish ~50 kB data on
        one nostr note, which is just json wrapper for any data content.
        See <a href="https://www.e2encrypted.com/nostr/nips/">nostr documentation</a>
        for more information.</p>

        <h2>immutable or updatable content</h2>
        <p>
        Datauris are by default immutable, ie. developer cannot update its content after its published,
        however this is not true for dynamically loaded content. Developer can use dynamic content loader
        to load full page source from nostr event. Below is an example.
        </p>
        <ul>
          <li>r = relay_uri</li>
          <li>k = pubkey</li>
          <li>f = d_tag</li>
        </ul>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=filepublish.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>
        <p>In this case, developer can publish new nostr event using same pubkey and same d tag value, which will override the previous content</p>
      </div>
    </main>
    <script>
    document.onclick = function(e){
      if(!e.target.classList.contains("data")){
        for(let datauri of document.querySelectorAll("p.data")){
          datauri.classList.remove("scroll")
        }
      }
    }

    for(let datauri of document.querySelectorAll("p.data")){
      datauri.onclick = function(){
        const range = document.createRange()
        range.selectNode(this)
        getSelection().removeAllRanges()
        getSelection().addRange(range)
        this.classList.add("scroll")
      }
    }

    function apply_hash(){
      const active_index = location.hash === "#devs" ? 1 : 0
      document.querySelector(".menu.active").classList.remove("active")
      document.querySelector(".content.active").classList.remove("active")
      document.querySelectorAll(".menu")[active_index].classList.add("active")
      document.querySelectorAll(".content")[active_index].classList.add("active")
    }

    addEventListener("hashchange", apply_hash)
    apply_hash()
    </script>
  </body>
</html>
#

#datauri_info #datauri #info

data:text/html;base64,<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <title>datauri info</title>
    <style>
      body {
        margin: 1em 1em 3em 1em;
      }

      header h1,
      header h2 {
        display: inline-block;
        margin-right: 1em;
      }

      .menu,
      .content {
        display: none;
      }

      .active {
        display: block;
      }

      h2 > a {
        padding: .2em .3em;
      }

      a:visited {
        color: inherit;
      }

      a:hover {
        color: rgb(200, 0, 0);
      }

      .data {
        background: rgb(220, 220, 100);
        /*background: rgb(180, 200, 230);*/
        /*background: rgb(180, 200, 230);*/
        overflow: hidden;
        word-break: break-all;
        max-height: 10em;
        padding: .3em;
      }

      .data,
      a {
        border-radius: .3em;
      }

      .data.scroll {
        overflow: scroll;
      }

      h2 {
        margin: 1em 0 .4em 0;
      }

      h2:first-child {
        margin-top: 0;
      }

      p {
        margin: 0 0 1em 0;
      }
    </style>
  </head>
  <body>
    <header>
      <div class="menu active">
        <h1>datauri info</h1>
        <h2>
          <a href="#devs">datauri for devs</a>
        </h2>
      </div>
      <div class="menu">
        <h1>datauri for devs</h1>
        <h2>
          <a href="#">datauri info</a>
        </h2>
      </div>
    </header>
    <main>
      <div class="content active">
        <p>
        What is datauri application? Apps hosted as datauris are basically portable web apps that work without http server.
        If you see this page, you probably already know how datauri apps work. You just copy-paste the link and access the app.
        Here are some datauri apps.
        </p>
        <h2>wired</h2>
        <p>Anon approved pow based nostr client</p>
        <p class="data">data:text/html;base64,PCFkb2N0eXBlIGh0bWw+PGh0bWwgbGFuZz0iZW4iPjxoZWFkPjxtZXRhIGNoYXJzZXQ9InV0Zi04Ii8+PG1ldGEgbmFtZT0idmlld3BvcnQiIGNvbnRlbnQ9IndpZHRoPWRldmljZS13aWR0aCxpbml0aWFsLXNjYWxlPTEiLz48bWV0YSBuYW1lPSJ0aGVtZS1jb2xvciIgY29udGVudD0iIzAwMDAwMCIvPjxtZXRhIG5hbWU9Im1vYmlsZS13ZWItYXBwLWNhcGFibGUiIGNvbnRlbnQ9InllcyIvPjxtZXRhIG5hbWU9ImFwcGxlLXRvdWNoLWZ1bGxzY3JlZW4iIGNvbnRlbnQ9InllcyIvPjxtZXRhIG5hbWU9ImFwcGxlLW1vYmlsZS13ZWItYXBwLXRpdGxlIiBjb250ZW50PSJFeHBvIi8+PG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+PG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtc3RhdHVzLWJhci1zdHlsZSIgY29udGVudD0iZGVmYXVsdCIvPjxtZXRhIG5hbWU9ImRlc2NyaXB0aW9uIiBjb250ZW50PSJUaGUgV2lyZWQiLz48dGl0bGU+VGhlIFdpcmVkPC90aXRsZT48L2hlYWQ+PGJvZHk+PG5vc2NyaXB0PllvdSBuZWVkIHRvIGVuYWJsZSBKYXZhU2NyaXB0IHRvIHJ1biB0aGlzIGFwcC48L25vc2NyaXB0PjxkaXYgaWQ9InJvb3QiPjwvZGl2PjxzY3JpcHQ+Cihhc3luYyBmdW5jdGlvbigpewogIGFzeW5jIGZ1bmN0aW9uIGRnZXQoaWQsIHJlbGF5KXsKICAgIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSkgPT4gewogICAgICBjb25zdCBzb2NrZXQgPSBuZXcgV2ViU29ja2V0KHJlbGF5KQogICAgICBsZXQgZmlsZV9kYXRhID0gW10KICAgICAgbGV0IGZpbGVpZHMgPSBbXQoKICAgICAgc29ja2V0Lm9ub3BlbiA9IGZ1bmN0aW9uKCl7CiAgICAgICAgc29ja2V0LnNlbmQoSlNPTi5zdHJpbmdpZnkoWyJSRVEiLCAiZmlsZWlkcyIsIHsiaWRzIjogW2lkXX1dKSkKICAgICAgfQoKICAgICAgc29ja2V0Lm9ubWVzc2FnZSA9IGFzeW5jIGZ1bmN0aW9uKGUpewogICAgICAgIGNvbnN0IGRhdGEgPSBKU09OLnBhcnNlKGUuZGF0YSkKCiAgICAgICAgaWYoZGF0YVswXSA9PT0gIkVWRU5UIiAmJiBkYXRhWzFdID09PSAiZmlsZWlkcyIpewogICAgICAgICAgZmlsZWlkcy5wdXNoKC4uLkpTT04ucGFyc2UoZGF0YVsyXS5jb250ZW50KS5maWxlaWRzKQogICAgICAgICAgc29ja2V0LnNlbmQoSlNPTi5zdHJpbmdpZnkoWyJSRVEiLCAicGFydHMiLCB7ImlkcyI6IGZpbGVpZHMsICJsaW1pdCI6IDMwMDB9XSkpCiAgICAgICAgfQogICAgICAgIGVsc2UgaWYoZGF0YVswXSA9PT0gIkVWRU5UIiAmJiBkYXRhWzFdID09PSAicGFydHMiKXsKICAgICAgICAgIGNvbnN0IG1hdGNoID0gZGF0YVsyXS5jb250ZW50Lm1hdGNoKC9ecGFydChcZHsxLDR9KTsoXGR7MSw0fSk7W2EtejAtOV0rOy8pCgogICAgICAgICAgaWYobWF0Y2gpewogICAgICAgICAgICBjb25zdCBwYXJ0X2RhdGEgPSBkYXRhWzJdLmNvbnRlbnQuc3Vic3RyKG1hdGNoWzBdLmxlbmd0aCkKICAgICAgICAgICAgZmlsZV9kYXRhLnB1c2goW3BhcnNlSW50KG1hdGNoWzFdLCAxMCksIHBhcnRfZGF0YV0pCiAgICAgICAgICB9ZWxzZXsKICAgICAgICAgICAgZmlsZV9kYXRhLnB1c2goWzEsIGRhdGFbMl0uY29udGVudF0pCiAgICAgICAgICB9CgogICAgICAgICAgaWYoZmlsZV9kYXRhLmxlbmd0aCA9PT0gZmlsZWlkcy5sZW5ndGgpewogICAgICAgICAgICBmaWxlX2RhdGEgPSBmaWxlX2RhdGEuc29ydCgoYSwgYikgPT4gYVswXSAtIGJbMF0pLm1hcChlID0+IGVbMV0pCiAgICAgICAgICAgIGNvbnN0IGRhdGF1cmkgPSAiZGF0YToiICsgZmlsZV9kYXRhLmpvaW4oIiIpCiAgICAgICAgICAgIGNvbnN0IGJsb2IgPSBhd2FpdCAoYXdhaXQgZmV0Y2goZGF0YXVyaSkpLmJsb2IoKQogICAgICAgICAgICByZXNvbHZlKGJsb2IpCiAgICAgICAgICB9CiAgICAgICAgfQogICAgICB9CiAgICB9KQogIH0KCiAgY29uc3Qgd29ya2VyX3NvdXJjZSA9IGF3YWl0IGRnZXQoImZlZWNlYTAxZmU2ZDkyMGU5ZjU2M2U5OWU1MjdhMmE1MjVlODQwOTUyN2ViMjc3Yzk0ZDJlOTA3MzRkYzRjNTAiLCAid3NzOi8vbm9zLmxvbCIpCiAgY29uc3Qgd29ya2VyXzU5Ml91cmwgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGF3YWl0IGRnZXQoImZlMGVhNjA2ODBiMTYwYjYyN2M0YmYzNThlMDk1YWZhMTcxNWJhMTlhNWI5YzkwOTYzMGQ1MmQzZTQzMjQ3MmMiLCAid3NzOi8vbm9zLmxvbCIpKQogIGNvbnN0IGljb25fcG5nX3VybCA9IFVSTC5jcmVhdGVPYmplY3RVUkwoYXdhaXQgZGdldCgiMDVkZTM0ZWE5MDkyMmVlOTQxMDUzYzQzYjUwMGRhYTE0ODRlMWY3MzM2M2ZjNWRhNzIyMzRiNDAzYjk3NzM3OCIsICJ3c3M6Ly9ub3MubG9sIikpCiAgY29uc3QgbWFpbl9qc19tYXBfdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChhd2FpdCBkZ2V0KCI1NWVkNDdiMjk0NzJkZTE4Y2I3MWQ0M2NjOGNmZmU3NTcwNzJmNDA3Y2ZlOGNlYjRkMDQyNDdiZGU0NDc1M2VlIiwgIndzczovL25vcy5sb2wiKSkKCiAgbGV0IHdvcmtlcl9zb3VyY2Vfc3RyID0gYXdhaXQgd29ya2VyX3NvdXJjZS50ZXh0KCkKICB3b3JrZXJfc291cmNlX3N0ciA9IHdvcmtlcl9zb3VyY2Vfc3RyLnJlcGxhY2UoJ2ltcG9ydFNjcmlwdHModC5wK3QudShyKSknLCAnaW1wb3J0U2NyaXB0cygiJyArIHdvcmtlcl81OTJfdXJsICsgJyIpJykKICBjb25zdCB3b3JrZXJfdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChuZXcgQmxvYihbd29ya2VyX3NvdXJjZV9zdHJdLCB7dHlwZTogInRleHQvamF2YXNjcmlwdCJ9KSkKCiAgY29uc3Qgb3JpZ2luYWxXb3JrZXJDb25zdHJ1Y3RvciA9IFdvcmtlcjsKCiAgV29ya2VyID0gZnVuY3Rpb24odXJsKSB7CiAgICB1cmwgPSB3b3JrZXJfdXJsCiAgICByZXR1cm4gbmV3IG9yaWdpbmFsV29ya2VyQ29uc3RydWN0b3IodXJsKTsKICB9CgogIFVSTCA9IGNsYXNzIE15VVJMIGV4dGVuZHMgVVJMIHsKICAgIGNvbnN0cnVjdG9yKGlucHV0KSB7CiAgICAgIHN1cGVyKGlucHV0LmluZGV4T2YoIi8iKSA9PT0gMCAmJiAobG9jYXRpb24ucHJvdG9jb2wgKyBsb2NhdGlvbi5wYXRobmFtZSArIGlucHV0KSB8fCBpbnB1dCk7CiAgICB9CiAgfQoKICBkYXRhTG9jYWxTdG9yYWdlID0gewogICAgZ2V0SXRlbTogZnVuY3Rpb24oKXt9LAogICAgc2V0SXRlbTogZnVuY3Rpb24oKXt9CiAgfQoKICBjb25zdCBzY3JpcHRzID0gWwogICAgIjE1NDNhMDcwYTQyZThlMGMwNTg2MTg0MGJkZTdlNWI2OTFjNGI1ZGVjNjYyNzQ1ODUyYzIyODNjZmVmZDhjYTUiCiAgXQoKICBjb25zdCBzdHlsZXMgPSBbCiAgICAiNjk5NjA3NzRjZTZkZmE4MzFhMzMwODczNDEyNWQ4MGRlMjY5YTg0ZjU5ODg4YTdkMzEyMTVlYTFhYzU1MTE1MiIKICBdCgogIGNvbnN0IHNjcmlwdF9pZCA9IHNjcmlwdHNbMF0KICBjb25zdCBzY3JpcHRfc291cmNlID0gYXdhaXQgZGdldChzY3JpcHRfaWQsICJ3c3M6Ly9ub3MubG9sIikKICBsZXQgc2NyaXB0X3NvdXJjZV9zdHIgPSBhd2FpdCBzY3JpcHRfc291cmNlLnRleHQoKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZSgiL2ljb24ucG5nIiwgaWNvbl9wbmdfdXJsKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZSgvKHNvdXJjZU1hcHBpbmdVUkw9KShtYWluXC5bYS16MC05XXs4fVwuanNcLm1hcCkvLCAiJDFodHRwOi8vbG9jYWxob3N0OjgwMDAvJDIiKQogIHNjcmlwdF9zb3VyY2Vfc3RyID0gc2NyaXB0X3NvdXJjZV9zdHIucmVwbGFjZUFsbCgibG9jYWxTdG9yYWdlIiwgImRhdGFMb2NhbFN0b3JhZ2UiKQoKICBjb25zdCBzY3JpcHRfZWwgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCJzY3JpcHQiKQogIHNjcmlwdF9lbC5zcmMgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKG5ldyBCbG9iKFtzY3JpcHRfc291cmNlX3N0cl0sIHt0eXBlOiAidGV4dC9qYXZhc2NyaXB0In0pKQogIGNvbnNvbGUubG9nKCJzY3JpcHRfZWwuc3JjIiwgc2NyaXB0X2VsLnNyYykKICBkb2N1bWVudC5ib2R5LmFwcGVuZChzY3JpcHRfZWwpCgogIGZvcihsZXQgc3R5bGVfaWQgb2Ygc3R5bGVzKXsKICAgIGNvbnN0IHN0eWxlX3NvdXJjZSA9IGF3YWl0IGRnZXQoc3R5bGVfaWQsICJ3c3M6Ly9ub3MubG9sIikKICAgIGNvbnN0IHN0eWxlX2VsID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgibGluayIpCiAgICBzdHlsZV9lbC5yZWwgPSAic3R5bGVzaGVldCIKICAgIHN0eWxlX2VsLmhyZWYgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGF3YWl0IHN0eWxlX3NvdXJjZSkKICAgIGRvY3VtZW50LmJvZHkuYXBwZW5kKHN0eWxlX2VsKQogIH0KfSkoKQo8L3NjcmlwdD4KPC9ib2R5PjwvaHRtbD4K</p>

        <h2>ourchan</h2>
        <p>Anon approved image board</p>
        <p class="data">data:text/html;r=nos.lol;k=2b8d50fcc974ffb652c0ab33edebb6a9ce72cd154591969592bc8b607a4ea1f2;f=ourchan;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>filepublish</h2>
        <p>Upload files</p>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=filepublish.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>fileloader</h2>
        <p>Download/view files</p>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=fileloader.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>

        <h2>video clip creator</h2>
        <p>Create and upload 1 second video clips</p>
        <p class="data">data:text/html;base64,<!DOCTYPE html>
<html>
  <head>
    <title>video clip creator</title>
    <style>
    body {
      margin: 1em;
    }
    h1 {
      margin-top: 0;
      font-size: 1.5em;
    }
    #log {
      overflow: auto;
      position: absolute;
      bottom: 1em;
      top: 11em;
      border: .3em solid #666;
      border-radius: .3em;
      right: 1em;
      left: 1em;
      padding: .5em;
      background: #eee;
    }
    #progress {
      background: #ddd;
      border-radius: .3em;
    }
    #progressbar {
      height: 1.3em;
      background: blue;
      width: 0%;
      border-radius: .4em;
    }
    #log > a {
      margin-bottom: 1em;
      display: block;
    }
    p {
      margin-top: 0;
      max-height: 7em;
      overflow: scroll;
    }
    video {
      display: none;
      z-index: 1;
    }
    #canvas,
    video {
      position: absolute;
      top: 1em;
      right: 1em;
      width: 10em;
      height: 6em;
    }
    </style>
  </head>
  <body>
    <h1>video clip creator</h1>
    <form id="form">
      <p>
        <input id="file" type="file"/>
      </p>
      <p>
        <input id="submit" type="submit" value="encode"/>
      </p>
    </form>
    <video id="video"></video>
    <video id="video2" controls loop="1"></video>
    <canvas id="canvas"></canvas>
    <div id="progress">
      <div id="progressbar"></div>
    </div>
    <div id="log"><div>
    <script type="module">
    const canvas = document.getElementById("canvas")
    const ctx = canvas.getContext('2d')
    const log = document.getElementById("log")
    const video = document.getElementById("video")
    const video2 = document.getElementById("video2")
    const progressbar = document.getElementById("progressbar")
    let files = null

    const frame_duration = 1/30
    const extract_frame_count = 30//1 / frame_duration
    let time = 15

    const buffer = await fetch(
      "https://unpkg.com/webm-wasm@0.4.1/dist/webm-worker.js"
    ).then(r => r.arrayBuffer())

    const worker = new Worker(
      URL.createObjectURL(new Blob([buffer], { type: "text/javascript" }))
    )

    let encoded_data = []
    let start_time = null
    let ready = false

    document.getElementById("form").onsubmit = function(e){
      e.preventDefault()
      video2.style.display = "none";
      video2.pause()
      progressbar.style.width = "0.5%"
      this.querySelectorAll("input").forEach(el => el.disabled = true)
      document.getElementById("submit").value = "encoding"
      files = document.getElementById("file").files
      start()
    }

    function blobToDataURL(blob, callback) {
      return new Promise(resolve => {
        var a = new FileReader()
        a.onload = function(e) {
          resolve(e.target.result)
        }
        a.readAsDataURL(blob)
      })
    }

    function info(msg, type){
      const el = document.createElement(type === 2 && "a" || "p")
      el[(type === 3) && "innerHTML" || "innerText"] = msg

      if(type){
        el.href = msg
      }

      log.append(el)
      log.scrollTo(0, 1e6)
    }

    async function process_video(){
      let frame_count = 0

      video.onseeked = function(e) {
        frame_count++
        info("extract frame " + frame_count + " / " + extract_frame_count)

        ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
        const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height)

        if(frame_count < extract_frame_count){
          time += frame_duration
          this.currentTime = Math.min(Math.max(0, (time < 0 ? this.duration : 0) + time), this.duration)
          worker.postMessage(imageData.data.buffer, [imageData.data.buffer])
        }else{
          worker.postMessage(null)
        }
      }
    }

    function total_time(start){
      return (new Date().getTime() - start) / 1000
    }

    function createBufferURL(buffer, type = '') {
      return URL.createObjectURL(new Blob([buffer], {type}));
    }

    async function show_data_uri(blob){
      let blob_url = URL.createObjectURL(blob)

      info(await blobToDataURL(blob))
      info(blob_url, 2)

      video2.src = blob_url
      video2.style.display = "block";
      video2.play()

      progressbar.style.width = "100%"
      document.getElementById("form").reset()
      document.getElementById("submit").value = "encode"
      document.querySelectorAll("form input").forEach(el => el.disabled = false)
    }

    worker.onmessage = function(e){
      if(!e.data) {
        return
      }

      if(e.data != null && typeof e.data == "object" && e.data.byteLength > 0){
        encoded_data.push(e.data)
        info("encode " + encoded_data.length + " / " + extract_frame_count)
        progressbar.style.width = parseInt((encoded_data.length / extract_frame_count) * 100, 10) + "%"
      }

      if(e.data == "READY"){
        info("ready", canvas.width, canvas.height)

        worker.postMessage({
          width: canvas.width,
          height: canvas.height,
          realtime: true,
          bitrate: 100
        })

        //for(let i = 0; i < frames.length; i++){
        //  worker.postMessage(frames[i], [frames[i]])
        //}

        ready = true
        process_video()
      }
      else if(ready){
        if(e.data.byteLength == 0){
          info("encoding finished (duration: " + parseInt(total_time(start_time), 10) +
            " s, speed: " + parseInt(total_time(start_time) / extract_frame_count, 10) +" s / frame)")
          let blob = new Blob(encoded_data, { type: 'video/webm' })
          show_data_uri(blob)
        }
      }
    }

    function load_video(){
      return new Promise(resolve => {
        video.src = URL.createObjectURL(files[0])

        video.onloadedmetadata = function() {
          canvas.height = video.videoHeight
          canvas.width = video.videoWidth
          this.currentTime = Math.min(Math.max(0, (time < 0 ? this.duration : 0) + time), this.duration)
          resolve()
        }
      })
    }

    async function start(){
      start_time = new Date().getTime()
      window.start_time = start_time
      await load_video()
      worker.postMessage("https://unpkg.com/webm-wasm@0.4.1/dist/webm-wasm.wasm")
    }
    </script>
  </body>
</html>
</p>
      </div>
      <div class="content">
        <h2>truly distributed, local first apps</h2>
        <p>
        Datauri apps dont rely on http server for initialization. Many apps however load dynamic content from internet.
        Releasing datauri apps is very simple, because there is no need for web server or domain name.
        </p>
        <p>Using nostr as backbone for dynamic content provides distributed alternative for centralized solutions.
        It is simple to load content from nostr notes. Nostr is based on websocket connection. You can publish ~50 kB data on
        one nostr note, which is just json wrapper for any data content.
        See <a href="https://www.e2encrypted.com/nostr/nips/">nostr documentation</a>
        for more information.</p>

        <h2>immutable or updatable content</h2>
        <p>
        Datauris are by default immutable, ie. developer cannot update its content after its published,
        however this is not true for dynamically loaded content. Developer can use dynamic content loader
        to load full page source from nostr event. Below is an example.
        </p>
        <ul>
          <li>r = relay_uri</li>
          <li>k = pubkey</li>
          <li>f = d_tag</li>
        </ul>
        <p class="data">data:text/html;r=nos.lol;k=01b0b1960ffe89eaa11e67ca999d832e55350dd17c0ea9eeee1d5d6ac6f0fdd4;f=filepublish.html;base64,PHNjcmlwdD4KKCgpID0+IHsKICBjb25zdCByID0gIndzczovLyIgKyBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87cj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGsgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87az0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQogIGNvbnN0IGYgPSBsb2NhdGlvbi5ocmVmLm1hdGNoKC9kYXRhOlxTKz87Zj0oW147XSspXFMqO2Jhc2U2NCwvKVsxXQoKICBkb2N1bWVudC53cml0ZSgiPHA+Y29ubmVjdGluZyB0byAiICsgciArICI8L3A+IikKICBjb25zdCBzID0gbmV3IFdlYlNvY2tldChyKQoKICBzLm9ub3BlbiA9ICgpID0+IHsKICAgIHMuc2VuZCgnWyJSRVEiLCAicSIsIHsiYXV0aG9ycyI6IFsiJyArIGsgKyAnIl0sICIjZCI6IFsiJyArIGYgKyAnIl19XScpCiAgfQoKICBzLm9ubWVzc2FnZSA9IGFzeW5jIChlKSA9PiB7CiAgICBzLmNsb3NlKCkKICAgIGRvY3VtZW50LndyaXRlKEpTT04ucGFyc2UoZS5kYXRhKVsyXS5jb250ZW50KQoKICAgIHdoaWxlKCF3aW5kb3cubG9hZGVkKXsKICAgICAgYXdhaXQgbmV3IFByb21pc2UoYyA9PiBzZXRUaW1lb3V0KGMsIDEwKSkKICAgIH0KCiAgICBkaXNwYXRjaEV2ZW50KG5ldyBDdXN0b21FdmVudCgibG9hZCIpKQogIH0KfSkoKQo8L3NjcmlwdD4K</p>
        <p>In this case, developer can publish new nostr event using same pubkey and same d tag value, which will override the previous content</p>
      </div>
    </main>
    <script>
    document.onclick = function(e){
      if(!e.target.classList.contains("data")){
        for(let datauri of document.querySelectorAll("p.data")){
          datauri.classList.remove("scroll")
        }
      }
    }

    for(let datauri of document.querySelectorAll("p.data")){
      datauri.onclick = function(){
        const range = document.createRange()
        range.selectNode(this)
        getSelection().removeAllRanges()
        getSelection().addRange(range)
        this.classList.add("scroll")
      }
    }

    function apply_hash(){
      const active_index = location.hash === "#devs" ? 1 : 0
      document.querySelector(".menu.active").classList.remove("active")
      document.querySelector(".content.active").classList.remove("active")
      document.querySelectorAll(".menu")[active_index].classList.add("active")
      document.querySelectorAll(".content")[active_index].classList.add("active")
    }

    addEventListener("hashchange", apply_hash)
    apply_hash()
    </script>
  </body>
</html>
#