From d57e6f8e097da6b22212ed550d6d2a0a6b7404f8 Mon Sep 17 00:00:00 2001 From: bobbydigitales Date: Sun, 10 Mar 2024 15:48:35 -0700 Subject: [PATCH] write TODO list and TypeScript map format has changed --- db.js | 2 +- main.go | 39 +++++++++++++++++++++++++++++++++++++++ main.js | 2 +- sw.js | 2 +- 4 files changed, 42 insertions(+), 3 deletions(-) diff --git a/db.js b/db.js index 8cc0cdc..e81f273 100644 --- a/db.js +++ b/db.js @@ -132,4 +132,4 @@ export async function getAllData(userID) { }; }); } -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"db.js","sourceRoot":"","sources":["src/db.ts"],"names":[],"mappings":"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","sourcesContent":["// interface MyJsonObject {\n//   id: string;\n//   name: string;\n//   email: string;\n// }\n\nconst dbName: string = \"ddln\";\nconst storeNameBase: string = \"posts\";\nlet keyBase = \"dandelion_posts_v1_\"\nlet key = \"\";\n\n\ninterface IDBRequestEvent<T = any> extends Event {\n    target: IDBRequest<T>;\n}\n\n// IndexedDB uses DOMException, so let's use it for error typing\ntype DBError = Event & {\n    target: { errorCode: DOMException };\n};\n\nexport function openDatabase(userID:string): Promise<IDBDatabase> {\n    const storeName = `${storeNameBase}_${userID}`;\n\n    return new Promise((resolve, reject) => {\n        const request: IDBOpenDBRequest = indexedDB.open(dbName, 1);\n\n        request.onerror = (event: Event) => {\n            // Use a type assertion to access the specific properties of IDBRequest error event\n            const errorEvent = event as IDBRequestEvent;\n            reject(`Database error: ${errorEvent.target.error?.message}`);\n        };\n\n        request.onupgradeneeded = (event: IDBVersionChangeEvent) => {\n            const db: IDBDatabase = (event.target as IDBOpenDBRequest).result;\n            if (!db.objectStoreNames.contains(storeName)) {\n                let store = db.createObjectStore(storeName, { keyPath: \"id\", autoIncrement: true });\n                store.createIndex(\"datetimeIndex\", \"post_timestamp\", { unique: false });\n\n            }\n        };\n\n        request.onsuccess = (event: Event) => {\n            const db: IDBDatabase = (event.target as IDBOpenDBRequest).result;\n            resolve(db);\n        };\n    });\n}\n\n\n\nexport async function addData(userID: string, data: any): Promise<void> {\n    try {\n        const storeName = `${storeNameBase}_${userID}`;\n        const db = await openDatabase(userID);\n        const transaction = db.transaction(storeName, \"readwrite\");\n        const store = transaction.objectStore(storeName);\n\n        const addRequest = store.add({post_timestamp: data.post_timestamp, data:data});\n\n        addRequest.onsuccess = (e: Event) => {\n            // console.log('Data has been added:', (e.target as IDBRequest).result);\n        };\n\n        addRequest.onerror = (event: Event) => {\n            // Use a type assertion to access the specific properties of IDBRequest error event\n            const errorEvent = event as IDBRequestEvent;\n            console.error('Error in adding data:', errorEvent.target.error?.message);\n        };\n    } catch (error) {\n        console.error('Error in opening database:', error);\n    }\n}\n\nexport async function addDataArray(userID: string, array: any[]): Promise<void> {\n    try {\n        const storeName = `${storeNameBase}_${userID}`;\n        const db = await openDatabase(userID);\n        const transaction = db.transaction(storeName, \"readwrite\");\n        const store = transaction.objectStore(storeName);\n\n        let count = 0;\n\n        array.reverse();\n\n        for (let data of array) {\n            const addRequest = store.add({post_timestamp: data.post_timestamp, data:data});\n            addRequest.onsuccess = (e: Event) => {\n                // console.log('Data has been added:', (e.target as IDBRequest).result);\n            };\n    \n            addRequest.onerror = (event: Event) => {\n                // Use a type assertion to access the specific properties of IDBRequest error event\n                const errorEvent = event as IDBRequestEvent;\n                console.error('Error in adding data:', errorEvent.target.error?.message);\n            };\n\n            count++;\n\n            // if (count % 100 === 0) {\n            //     console.log(`Added ${count} posts...`);\n            // }\n        }\n\n       \n    } catch (error) {\n        console.error('Error in opening database:', error);\n    }\n}\n\nexport async function getData(userID:string, lowerID:Date, upperID:Date): Promise<any | undefined> {\n    const storeName = `${storeNameBase}_${userID}`;\n    const db = await openDatabase(userID);\n    const transaction = db.transaction(storeName, \"readonly\");\n    const store = transaction.objectStore(storeName);\n\n    return new Promise((resolve, reject) => {\n        const keyRangeValue = IDBKeyRange.bound(lowerID, upperID);\n\n        const records: any[] = [];\n\n        const index = store.index(\"datetimeIndex\");\n\n\n        const cursorRequest = index.openCursor(keyRangeValue);\n\n        cursorRequest.onsuccess = (event: Event) => {\n            const cursor = (event.target as IDBRequest).result as IDBCursorWithValue;\n            if (cursor) {\n                records.push(cursor.value.data); // Collect the record\n                cursor.continue(); // Move to the next item in the range\n            } else {\n                // No more entries in the range\n                resolve(records);\n            }\n        };\n\n        cursorRequest.onerror = (event: Event) => {\n            // Use a type assertion to access the specific properties of IDBRequest error event\n            const errorEvent = event as IDBRequestEvent;\n            console.error('Transaction failed:', errorEvent.target.error?.message);\n            reject(errorEvent.target.error); // Reject the promise if there's an error\n        };\n    });\n}\n\nexport async function getAllData(userID:string): Promise<any | undefined> {\n    const storeName = `${storeNameBase}_${userID}`;\n    const db = await openDatabase(userID);\n    const transaction = db.transaction(storeName, \"readonly\");\n    const store = transaction.objectStore(storeName);\n\n    return new Promise((resolve, reject) => {\n        const getRequest = store.getAll();\n\n        getRequest.onsuccess = () => {\n            if (getRequest.result) {\n                //   console.log('Retrieved data:', getRequest.result.jsonData);\n                // resolve(getRequest.result.jsonData as any);\n                resolve(getRequest.result);\n            } else {\n                console.log('No data record found for key', key);\n                resolve(undefined); // explicitly resolve with undefined when no data is found\n            }\n        };\n\n        getRequest.onerror = (event: Event) => {\n            // Use a type assertion to access the specific properties of IDBRequest error event\n            const errorEvent = event as IDBRequestEvent;\n            console.error('Transaction failed:', errorEvent.target.error?.message);\n            reject(errorEvent.target.error); // reject the promise if there's an error\n        };\n    });\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"db.js","sourceRoot":"","sources":["src/db.ts"],"names":[],"mappings":"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","sourcesContent":["// interface MyJsonObject {\n//   id: string;\n//   name: string;\n//   email: string;\n// }\n\nconst dbName: string = \"ddln\";\nconst storeNameBase: string = \"posts\";\nlet keyBase = \"dandelion_posts_v1_\"\nlet key = \"\";\n\n\ninterface IDBRequestEvent<T = any> extends Event {\n    target: IDBRequest<T>;\n}\n\n// IndexedDB uses DOMException, so let's use it for error typing\ntype DBError = Event & {\n    target: { errorCode: DOMException };\n};\n\nexport function openDatabase(userID:string): Promise<IDBDatabase> {\n    const storeName = `${storeNameBase}_${userID}`;\n\n    return new Promise((resolve, reject) => {\n        const request: IDBOpenDBRequest = indexedDB.open(dbName, 1);\n\n        request.onerror = (event: Event) => {\n            // Use a type assertion to access the specific properties of IDBRequest error event\n            const errorEvent = event as IDBRequestEvent;\n            reject(`Database error: ${errorEvent.target.error?.message}`);\n        };\n\n        request.onupgradeneeded = (event: IDBVersionChangeEvent) => {\n            const db: IDBDatabase = (event.target as IDBOpenDBRequest).result;\n            if (!db.objectStoreNames.contains(storeName)) {\n                let store = db.createObjectStore(storeName, { keyPath: \"id\", autoIncrement: true });\n                store.createIndex(\"datetimeIndex\", \"post_timestamp\", { unique: false });\n\n            }\n        };\n\n        request.onsuccess = (event: Event) => {\n            const db: IDBDatabase = (event.target as IDBOpenDBRequest).result;\n            resolve(db);\n        };\n    });\n}\n\n\n\nexport async function addData(userID: string, data: any): Promise<void> {\n    try {\n        const storeName = `${storeNameBase}_${userID}`;\n        const db = await openDatabase(userID);\n        const transaction = db.transaction(storeName, \"readwrite\");\n        const store = transaction.objectStore(storeName);\n\n        const addRequest = store.add({post_timestamp: data.post_timestamp, data:data});\n\n        addRequest.onsuccess = (e: Event) => {\n            // console.log('Data has been added:', (e.target as IDBRequest).result);\n        };\n\n        addRequest.onerror = (event: Event) => {\n            // Use a type assertion to access the specific properties of IDBRequest error event\n            const errorEvent = event as IDBRequestEvent;\n            console.error('Error in adding data:', errorEvent.target.error?.message);\n        };\n    } catch (error) {\n        console.error('Error in opening database:', error);\n    }\n}\n\nexport async function addDataArray(userID: string, array: any[]): Promise<void> {\n    try {\n        const storeName = `${storeNameBase}_${userID}`;\n        const db = await openDatabase(userID);\n        const transaction = db.transaction(storeName, \"readwrite\");\n        const store = transaction.objectStore(storeName);\n\n        let count = 0;\n\n        array.reverse();\n\n        for (let data of array) {\n            const addRequest = store.add({post_timestamp: data.post_timestamp, data:data});\n            addRequest.onsuccess = (e: Event) => {\n                // console.log('Data has been added:', (e.target as IDBRequest).result);\n            };\n    \n            addRequest.onerror = (event: Event) => {\n                // Use a type assertion to access the specific properties of IDBRequest error event\n                const errorEvent = event as IDBRequestEvent;\n                console.error('Error in adding data:', errorEvent.target.error?.message);\n            };\n\n            count++;\n\n            // if (count % 100 === 0) {\n            //     console.log(`Added ${count} posts...`);\n            // }\n        }\n\n       \n    } catch (error) {\n        console.error('Error in opening database:', error);\n    }\n}\n\nexport async function getData(userID:string, lowerID:Date, upperID:Date): Promise<any | undefined> {\n    const storeName = `${storeNameBase}_${userID}`;\n    const db = await openDatabase(userID);\n    const transaction = db.transaction(storeName, \"readonly\");\n    const store = transaction.objectStore(storeName);\n\n    return new Promise((resolve, reject) => {\n        const keyRangeValue = IDBKeyRange.bound(lowerID, upperID);\n\n        const records: any[] = [];\n\n        const index = store.index(\"datetimeIndex\");\n\n\n        const cursorRequest = index.openCursor(keyRangeValue);\n\n        cursorRequest.onsuccess = (event: Event) => {\n            const cursor = (event.target as IDBRequest).result as IDBCursorWithValue;\n            if (cursor) {\n                records.push(cursor.value.data); // Collect the record\n                cursor.continue(); // Move to the next item in the range\n            } else {\n                // No more entries in the range\n                resolve(records);\n            }\n        };\n\n        cursorRequest.onerror = (event: Event) => {\n            // Use a type assertion to access the specific properties of IDBRequest error event\n            const errorEvent = event as IDBRequestEvent;\n            console.error('Transaction failed:', errorEvent.target.error?.message);\n            reject(errorEvent.target.error); // Reject the promise if there's an error\n        };\n    });\n}\n\nexport async function getAllData(userID:string): Promise<any | undefined> {\n    const storeName = `${storeNameBase}_${userID}`;\n    const db = await openDatabase(userID);\n    const transaction = db.transaction(storeName, \"readonly\");\n    const store = transaction.objectStore(storeName);\n\n    return new Promise((resolve, reject) => {\n        const getRequest = store.getAll();\n\n        getRequest.onsuccess = () => {\n            if (getRequest.result) {\n                //   console.log('Retrieved data:', getRequest.result.jsonData);\n                // resolve(getRequest.result.jsonData as any);\n                resolve(getRequest.result);\n            } else {\n                console.log('No data record found for key', key);\n                resolve(undefined); // explicitly resolve with undefined when no data is found\n            }\n        };\n\n        getRequest.onerror = (event: Event) => {\n            // Use a type assertion to access the specific properties of IDBRequest error event\n            const errorEvent = event as IDBRequestEvent;\n            console.error('Transaction failed:', errorEvent.target.error?.message);\n            reject(errorEvent.target.error); // reject the promise if there's an error\n        };\n    });\n}\n"]} \ No newline at end of file diff --git a/main.go b/main.go index 8dc7d07..4c2c9d7 100644 --- a/main.go +++ b/main.go @@ -1,3 +1,41 @@ +// TODO +// 1. Convery network messages to flatbuffers so we don't need to parse JSON in Go. +// Use binary WS messages. Generate for Go and JS and use that everywhere. We can +// also use this for WebRTC messages. +// 2. Keep a list of all nodes that bootstrapped in the last N minutes. +// 3. When a node bootstraps, send it a random list of the nodes we know about. +// 4. Do the signalling to connect those nodes. +// 5. Each node will know about N other nodes. To find get the data for a feed, +// a node will need to find another node that has that data. To do that we'll need to +// implement a search message that is sent to all currently connected nodes, and they +// forward to all their nodes, passing back the address of the node that has the data. +// Once we find it, we'll do the signalling to connect to it via Web RTC via our existing connected nodes. +// ----- +// Feeds. People can curate feeds which can be any combination of hashtags, serch terms and users. +// Invite-only communities. Just block everyone else even if they post to it. +// Limit to friends and friends of friends + +// MVP +// You connect to the person you want to get the post from to get the post +// they give you the post +// If they're offline, you can't get their updates. +// This is very stupid, but it's simplest thing. +// The bootstrap server connects you to them directly via WebRTC +// This will make the thing actually function as a little toy for people to play with. +// This will let us test whether background tabs respond to webrtc requests. + +// THEN +// Need to have identity sorted out +// When you read someone's posts, you also cache them locally +// cache priority goes mutuals->people you follow->people who you folllow, follow, so you're always +// caching your mutual's posts +// Posts are samll, so caching per-post will work fine. +// Then the process is for the bootstrap server to remember all nodes and what they're caching +// This will allow distributed content delivery but put a memory and bendwidth strain on the +// bootstrap sever. Look into Web Transport for the raspberry pi overhead. Could buy a few more RPIs +// and make a little cluster +// Domain name so we can get a certificate and serve HTTPS / HTTP3 + package main import ( @@ -34,6 +72,7 @@ func handleWebSocket(w http.ResponseWriter, r *http.Request) { break } log.Printf("recv: %s", message) + err = conn.WriteMessage(mt, message) if err != nil { log.Println("Write error:", err) diff --git a/main.js b/main.js index 4a4592a..672649e 100644 --- a/main.js +++ b/main.js @@ -348,4 +348,4 @@ function renderPost(post) { return containerDiv; } window.addEventListener("load", main); -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"main.js","sourceRoot":"","sources":["src/main.ts"],"names":[],"mappings":"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","sourcesContent":["import { openDatabase, getData, addData, addDataArray } from \"./db.js\"\n\ndeclare let marked: any;\n// let posts:any;\n// let keyBase = \"dandelion_posts_v1_\"\n// let key:string = \"\";\n\n// interface PostTimestamp {\n//   year: number,\n//   month: number,\n//   day: number,\n//   hour: number,\n//   minute: number,\n//   second: number,\n// }\n\nclass Post {\n  post_timestamp: Date;\n  author: string;\n  author_id: string;\n  text: string;\n  image_data: ArrayBuffer | null;\n\n\n  importedFrom: \"twitter\" | null;\n  importSource: any;\n\n  constructor(author: string, author_id: string, text: string, post_timestamp: Date, imageData: ArrayBuffer | null = null, importedFrom: \"twitter\" | null = null, importSource: any = null) {\n    this.post_timestamp = post_timestamp;\n\n    this.author = author;\n    this.author_id = author_id;\n    this.text = text;\n    this.image_data = imageData;\n\n    this.importedFrom = importedFrom;\n    this.importSource = importSource;\n  }\n}\n\nwindow.addEventListener('scroll', () => {\n  // Total height of the document\n  const totalPageHeight = document.body.scrollHeight;\n\n  // Current scroll position\n  const scrollPoint = window.scrollY + window.innerHeight;\n\n  // Check if scrolled to bottom\n  if (scrollPoint >= totalPageHeight) {\n    console.log('Scrolled to the bottom!');\n    console.log(scrollPoint, totalPageHeight);\n    // You can perform your action here\n  }\n});\n\n\nfunction initMarkdown() {\n  const renderer = new marked.Renderer();\n  renderer.link = (href: any, title: string, text: string) => {\n    return `<a href=\"${href}\" target=\"_blank\"${title ? ` title=\"${title}\"` : ''}>${text}</a>`;\n  };\n  marked.setOptions({ renderer: renderer });\n}\n\nfunction waitMs(durationMs: number) {\n  return new Promise(resolve => setTimeout(resolve, durationMs));\n}\n\nfunction uuidv4() {\n  return \"10000000-1000-4000-8000-100000000000\".replace(/[018]/g, (c: any) =>\n    (c ^ crypto.getRandomValues(new Uint8Array(1))[0] & 15 >> c / 4).toString(16)\n  );\n}\n\nlet logLines: string[] = [];\nlet logLength = 10;\nfunction log(message: string) {\n  logLines.push(`${new Date().toLocaleTimeString()}: ${message}`);\n  if (logLines.length > 10) {\n    logLines = logLines.slice(logLines.length - logLength);\n  }\n  let log = document.getElementById(\"log\");\n  if (!log) {\n    throw new Error();\n  }\n  log.innerText = logLines.join(\"\\n\");\n}\n\nfunction arrayBufferToBase64(buffer: ArrayBuffer) {\n  return new Promise((resolve, reject) => {\n    const blob = new Blob([buffer], { type: 'application/octet-stream' });\n    const reader = new FileReader();\n\n    reader.onloadend = () => {\n      const dataUrl = reader.result as string;\n      if (!dataUrl) {\n        resolve(null);\n        return;\n      }\n      const base64 = dataUrl.split(',')[1];\n      resolve(base64);\n    };\n\n    reader.onerror = (error) => {\n      reject(error);\n    };\n\n    reader.readAsDataURL(blob);\n  });\n}\n\nasync function createTestData() {\n  let postsTestData = await (await fetch(\"./postsTestData.json\")).json();\n\n  return postsTestData;\n}\n\nlet time = 0;\n\nfunction timerStart() {\n  time = performance.now();\n}\n\nfunction timerDelta() {\n  return performance.now() - time;\n}\n\nasync function getFixedTweetText(entry: any) {\n\n\n  let fullText = entry.tweet.full_text;\n\n  let linkMarkdown = \"\";\n  for (const url of entry.tweet.entities.urls) {\n    linkMarkdown = `[${url.display_url}](${url.expanded_url})`;\n    fullText = fullText.replace(url.url, linkMarkdown);\n  }\n\n  return fullText\n}\n\nasync function createTestData2(userID: string) {\n  log(\"Importing tweet archive\")\n  let postsTestData: any[] = [];\n\n  let response = await fetch(\"./tweets.js\");\n  let tweetsText = await response.text();\n  tweetsText = tweetsText.replace(\"window.YTD.tweets.part0\", \"window.tweetData\");\n\n  new Function(tweetsText)();\n\n\n  // let tweets = JSON.parse(tweetJSON);\n  let count = 0;\n\n  for (let entry of (window as any).tweetData) {\n    // if (entry.tweet.hasOwnProperty(\"in_reply_to_screen_name\") || entry.tweet.retweeted || entry.tweet.full_text.startsWith(\"RT\")) {\n    //   continue;\n    // }\n\n    let mediaURL: string = entry.tweet?.entities?.media?.[0]?.media_url;\n    let isImage = false;\n    if (mediaURL) {\n      isImage = mediaURL.includes('jpg');\n    }\n\n    let imageData = null;\n    if (isImage) {\n      try {\n        let response = await fetch(mediaURL);\n        await waitMs(100);\n        if (response.status === 200) {\n          imageData = await response.arrayBuffer();\n        }\n        console.log(imageData);\n      } catch (e) {\n        console.log(e);\n      }\n\n    }\n\n    let timeStamp = new Date(entry.tweet.created_at);\n    let tweetText = await getFixedTweetText(entry);\n    let newPost = new Post('bobbydigitales', userID, tweetText, timeStamp, imageData, 'twitter', entry);\n\n    postsTestData.push(newPost);\n\n    count++;\n    if (count % 100 === 0) {\n      log(`Imported ${count} posts...`);\n      // render(postsTestData);\n    }\n\n    // if (count == 100-1) {\n    //   break;\n    // }\n\n  }\n  return postsTestData;\n}\n\nasync function createTestData3(userID: string) {\n  let posts = await (await (fetch('./posts.json'))).json();\n\n  return posts;\n}\n\nasync function registerServiceWorker() {\n  if (!(\"serviceWorker\" in navigator)) {\n    return;\n  }\n\n  let registrations = await navigator.serviceWorker.getRegistrations();\n  if (registrations.length > 0) {\n    console.log(\"Service worker already registered.\");\n    return registrations[0];\n  }\n\n  navigator.serviceWorker\n    .register(\"/sw.js\")\n    .then((registration) => {\n      console.log(\"Service Worker registered with scope:\", registration.scope);\n      return registration;\n    })\n    .catch((error) => {\n      console.error(\"Service Worker registration failed:\", error);\n    });\n}\n\nfunction addPost(userID: string, posts: Post[], postText: string) {\n  if ((typeof postText !== \"string\") || postText.length === 0) {\n    log(\"Not posting an empty string...\")\n    return;\n  }\n\n  let post = new Post(`bobbydigitales`, userID, postText, new Date());\n\n  posts.push(post);\n  // localStorage.setItem(key, JSON.stringify(posts));\n  addData(userID, post)\n\n  render(posts);\n}\n\nfunction generateID() {\n  if (self.crypto.hasOwnProperty(\"randomUUID\")) {\n    return self.crypto.randomUUID();\n  }\n\n  return uuidv4();\n}\n\nfunction getUserID() {\n  let id = localStorage.getItem(\"dandelion_id\");\n\n  if (!id) {\n    id = generateID();\n    localStorage.setItem(\"dandelion_id\", id);\n  }\n\n  return id;\n}\n\nfunction connectWebsocket(userID: string) {\n  let websocket = new WebSocket(`ws://${window.location.hostname}:${window.location.port}/ws`);\n\n  websocket.onopen = function (evt) {\n    log(\"Websocket: CONNECTED\");\n    websocket.send(`{\"messageType\":\"connect\", \"id\": \"${userID}\"}`);\n\n    let websocketPingInterval = window.setInterval(()=>{websocket.send(`{\"messageType\":\"ping\", \"id\": \"${userID}\"}`);}, 5000)\n  };\n\n  websocket.onclose = function (evt) {\n    log(\"Websocket: DISCONNECTED\");\n  };\n\n  websocket.onmessage = function (evt) {\n    log('Websocket: RESPONSE: ' + evt.data);\n  };\n\n  websocket.onerror = function (evt) {\n    log('Websocket: ERROR: ' + evt);\n  };\n\n  return websocket;\n}\n\nfunction setFont(fontName: string, fontSize: string) {\n\n  let content = document.getElementById('content');\n\n  if (!content) {\n    return;\n  }\n\n  content.style.fontFamily = fontName;\n  content.style.fontSize = fontSize;\n\n  let textArea = document.getElementById('textarea_post');\n  if (!textArea) {\n    return;\n  }\n\n  textArea.style.fontFamily = fontName;\n  textArea.style.fontSize = fontSize;\n}\n\nfunction initOffline() {\n  // Event listener for going offline\n  window.addEventListener('offline', () => { log(\"offline\") });\n\n  // Event listener for going online\n  window.addEventListener('online', () => { log(\"online\") });\n\n  log(`Online status: ${navigator.onLine ? \"online\" : \"offline\"}`)\n\n}\n\nfunction initButtons(userID: string, posts: Post[]) {\n  let font1Button = document.getElementById(\"button_font1\") as HTMLButtonElement;\n  let font2Button = document.getElementById(\"button_font2\") as HTMLButtonElement;\n\n  font1Button.addEventListener('click', () => { setFont('Bookerly', '16px') });\n  font2Button.addEventListener('click', () => { setFont('Virgil', '24px') });\n\n\n  let postButton = document.getElementById(\"button_post\") as HTMLButtonElement;\n  let postText = document.getElementById(\"textarea_post\") as HTMLTextAreaElement;\n\n  if (!(postButton && postText)) {\n    throw new Error();\n  }\n\n  postButton.addEventListener(\"click\", () => {\n    addPost(userID, posts, postText.value);\n    postText.value = \"\";\n  });\n\n}\n\nasync function loadPosts(userID: string) {\n\n  timerStart();\n  let posts: any = await getData(userID, new Date(2022, 8), new Date());\n\n  if (posts.length > 0) {\n    log(`Loaded ${posts.length} posts in ${timerDelta().toFixed(2)}ms`);\n    return posts;\n  }\n\n  posts = await createTestData2(userID);\n\n  log(\"Adding test data...\");\n  addDataArray(userID, posts);\n  return await getData(userID, new Date(2022, 8), new Date());\n}\n\nasync function main() {\n  let posts: Post[] = [];\n  let time = 0; ``\n  let delta = 0;\n\n  let urlParams = (new URL(window.location.href)).searchParams;\n\n  if (urlParams.get(\"sw\") === \"true\") {\n    let registration = await registerServiceWorker();\n  }\n\n  let userID = getUserID();\n  log(`Your user ID is: ${userID}`);\n\n  if (navigator.storage && navigator.storage.persist && !navigator.storage.persisted) {\n    const isPersisted = await navigator.storage.persist();\n    log(`Persisted storage granted: ${isPersisted}`);\n  }\n\n  log(`Persisted: ${(await navigator?.storage?.persisted())?.toString()}`);\n\n  initMarkdown();\n\n  // let main = await fetch(\"/main.js\");\n  // let code = await main.text();\n  //   console.log(code);\n  //   registration.active.postMessage({type:\"updateMain\", code:code});\n\n  posts = await loadPosts(userID);\n\n  let websocket = connectWebsocket(userID);\n  initOffline();\n  initButtons(userID, posts);\n  // debugger;\n\n  timerStart();\n  render(posts);\n  let renderTime = timerDelta();\n\n  log(`render took: ${renderTime.toFixed(2)}ms`);\n\n  log(`memory used: ${((performance as any).memory.usedJSHeapSize / 1024 / 1024).toFixed(2)}Mb`)\n}\n\nfunction render(posts: Post[]) {\n  const fragment = document.createDocumentFragment();\n  let contentDiv = document.getElementById(\"content\");\n  if (!contentDiv) {\n    throw new Error();\n  }\n  contentDiv.innerHTML = \"\";\n  let count = 0;\n\n\n  for (let i = posts.length - 1; i >= 0; i--) {\n    let postData = posts[i];\n\n    let post = renderPost(postData);\n\n    if (post) {\n      fragment.appendChild(post);\n      count++;\n    }\n    if (count > 100) {\n      break;\n    }\n  }\n\n\n  if (!contentDiv) {\n    throw new Error(\"Couldn't get content div!\");\n  }\n\n  contentDiv.appendChild(fragment);\n\n}\n\nfunction renderPost(post: Post) {\n  if (!(post.hasOwnProperty(\"text\"))) {\n    throw new Error(\"Post is malformed!\");\n  }\n  let containerDiv = document.createElement(\"div\");\n  let textDiv = document.createElement(\"div\");\n  let timestampDiv = document.createElement(\"div\");\n  let hr = document.createElement(\"hr\");\n\n\n  textDiv.innerHTML = marked.parse(post.text);\n  // textDiv.innerHTML = DOMPurify.sanitize(marked.parse(post.text));\n\n  timestampDiv.innerText = `${post.post_timestamp.toDateString()}`;\n  timestampDiv.title = `${post.post_timestamp.toLocaleTimeString()} · ${post.post_timestamp.toDateString()}`\n  containerDiv.appendChild(hr);\n  containerDiv.appendChild(textDiv);\n\n  if (!(\"image_data\" in post && post.image_data)) {\n    containerDiv.appendChild(timestampDiv);\n    return containerDiv;\n    // return null;\n  }\n\n  let image = document.createElement(\"img\");\n  const blob = new Blob([post.image_data as ArrayBuffer], { type: 'image/jpg' });\n  const url = URL.createObjectURL(blob);\n  image.onload = () => {\n    URL.revokeObjectURL(url);\n  };\n\n\n  image.src = url;\n  // image.src = image.src = \"data:image/png;base64,\" + post.image;\n  image.className = \"postImage\";\n\n  containerDiv.appendChild(image);\n  containerDiv.appendChild(timestampDiv);\n\n  return containerDiv;\n}\n\nwindow.addEventListener(\"load\", main);\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"main.js","sourceRoot":"","sources":["src/main.ts"],"names":[],"mappings":"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","sourcesContent":["import { openDatabase, getData, addData, addDataArray } from \"./db.js\"\n\ndeclare let marked: any;\n// let posts:any;\n// let keyBase = \"dandelion_posts_v1_\"\n// let key:string = \"\";\n\n// interface PostTimestamp {\n//   year: number,\n//   month: number,\n//   day: number,\n//   hour: number,\n//   minute: number,\n//   second: number,\n// }\n\nclass Post {\n  post_timestamp: Date;\n  author: string;\n  author_id: string;\n  text: string;\n  image_data: ArrayBuffer | null;\n\n\n  importedFrom: \"twitter\" | null;\n  importSource: any;\n\n  constructor(author: string, author_id: string, text: string, post_timestamp: Date, imageData: ArrayBuffer | null = null, importedFrom: \"twitter\" | null = null, importSource: any = null) {\n    this.post_timestamp = post_timestamp;\n\n    this.author = author;\n    this.author_id = author_id;\n    this.text = text;\n    this.image_data = imageData;\n\n    this.importedFrom = importedFrom;\n    this.importSource = importSource;\n  }\n}\n\nwindow.addEventListener('scroll', () => {\n  // Total height of the document\n  const totalPageHeight = document.body.scrollHeight;\n\n  // Current scroll position\n  const scrollPoint = window.scrollY + window.innerHeight;\n\n  // Check if scrolled to bottom\n  if (scrollPoint >= totalPageHeight) {\n    console.log('Scrolled to the bottom!');\n    console.log(scrollPoint, totalPageHeight);\n    // You can perform your action here\n  }\n});\n\n\nfunction initMarkdown() {\n  const renderer = new marked.Renderer();\n  renderer.link = (href: any, title: string, text: string) => {\n    return `<a href=\"${href}\" target=\"_blank\"${title ? ` title=\"${title}\"` : ''}>${text}</a>`;\n  };\n  marked.setOptions({ renderer: renderer });\n}\n\nfunction waitMs(durationMs: number) {\n  return new Promise(resolve => setTimeout(resolve, durationMs));\n}\n\nfunction uuidv4() {\n  return \"10000000-1000-4000-8000-100000000000\".replace(/[018]/g, (c: any) =>\n    (c ^ crypto.getRandomValues(new Uint8Array(1))[0] & 15 >> c / 4).toString(16)\n  );\n}\n\nlet logLines: string[] = [];\nlet logLength = 10;\nfunction log(message: string) {\n  logLines.push(`${new Date().toLocaleTimeString()}: ${message}`);\n  if (logLines.length > 10) {\n    logLines = logLines.slice(logLines.length - logLength);\n  }\n  let log = document.getElementById(\"log\");\n  if (!log) {\n    throw new Error();\n  }\n  log.innerText = logLines.join(\"\\n\");\n}\n\nfunction arrayBufferToBase64(buffer: ArrayBuffer) {\n  return new Promise((resolve, reject) => {\n    const blob = new Blob([buffer], { type: 'application/octet-stream' });\n    const reader = new FileReader();\n\n    reader.onloadend = () => {\n      const dataUrl = reader.result as string;\n      if (!dataUrl) {\n        resolve(null);\n        return;\n      }\n      const base64 = dataUrl.split(',')[1];\n      resolve(base64);\n    };\n\n    reader.onerror = (error) => {\n      reject(error);\n    };\n\n    reader.readAsDataURL(blob);\n  });\n}\n\nasync function createTestData() {\n  let postsTestData = await (await fetch(\"./postsTestData.json\")).json();\n\n  return postsTestData;\n}\n\nlet time = 0;\n\nfunction timerStart() {\n  time = performance.now();\n}\n\nfunction timerDelta() {\n  return performance.now() - time;\n}\n\nasync function getFixedTweetText(entry: any) {\n\n\n  let fullText = entry.tweet.full_text;\n\n  let linkMarkdown = \"\";\n  for (const url of entry.tweet.entities.urls) {\n    linkMarkdown = `[${url.display_url}](${url.expanded_url})`;\n    fullText = fullText.replace(url.url, linkMarkdown);\n  }\n\n  return fullText\n}\n\nasync function createTestData2(userID: string) {\n  log(\"Importing tweet archive\")\n  let postsTestData: any[] = [];\n\n  let response = await fetch(\"./tweets.js\");\n  let tweetsText = await response.text();\n  tweetsText = tweetsText.replace(\"window.YTD.tweets.part0\", \"window.tweetData\");\n\n  new Function(tweetsText)();\n\n\n  // let tweets = JSON.parse(tweetJSON);\n  let count = 0;\n\n  for (let entry of (window as any).tweetData) {\n    // if (entry.tweet.hasOwnProperty(\"in_reply_to_screen_name\") || entry.tweet.retweeted || entry.tweet.full_text.startsWith(\"RT\")) {\n    //   continue;\n    // }\n\n    let mediaURL: string = entry.tweet?.entities?.media?.[0]?.media_url;\n    let isImage = false;\n    if (mediaURL) {\n      isImage = mediaURL.includes('jpg');\n    }\n\n    let imageData = null;\n    if (isImage) {\n      try {\n        let response = await fetch(mediaURL);\n        await waitMs(100);\n        if (response.status === 200) {\n          imageData = await response.arrayBuffer();\n        }\n        console.log(imageData);\n      } catch (e) {\n        console.log(e);\n      }\n\n    }\n\n    let timeStamp = new Date(entry.tweet.created_at);\n    let tweetText = await getFixedTweetText(entry);\n    let newPost = new Post('bobbydigitales', userID, tweetText, timeStamp, imageData, 'twitter', entry);\n\n    postsTestData.push(newPost);\n\n    count++;\n    if (count % 100 === 0) {\n      log(`Imported ${count} posts...`);\n      // render(postsTestData);\n    }\n\n    // if (count == 100-1) {\n    //   break;\n    // }\n\n  }\n  return postsTestData;\n}\n\nasync function createTestData3(userID: string) {\n  let posts = await (await (fetch('./posts.json'))).json();\n\n  return posts;\n}\n\nasync function registerServiceWorker() {\n  if (!(\"serviceWorker\" in navigator)) {\n    return;\n  }\n\n  let registrations = await navigator.serviceWorker.getRegistrations();\n  if (registrations.length > 0) {\n    console.log(\"Service worker already registered.\");\n    return registrations[0];\n  }\n\n  navigator.serviceWorker\n    .register(\"/sw.js\")\n    .then((registration) => {\n      console.log(\"Service Worker registered with scope:\", registration.scope);\n      return registration;\n    })\n    .catch((error) => {\n      console.error(\"Service Worker registration failed:\", error);\n    });\n}\n\nfunction addPost(userID: string, posts: Post[], postText: string) {\n  if ((typeof postText !== \"string\") || postText.length === 0) {\n    log(\"Not posting an empty string...\")\n    return;\n  }\n\n  let post = new Post(`bobbydigitales`, userID, postText, new Date());\n\n  posts.push(post);\n  // localStorage.setItem(key, JSON.stringify(posts));\n  addData(userID, post)\n\n  render(posts);\n}\n\nfunction generateID() {\n  if (self.crypto.hasOwnProperty(\"randomUUID\")) {\n    return self.crypto.randomUUID();\n  }\n\n  return uuidv4();\n}\n\nfunction getUserID() {\n  let id = localStorage.getItem(\"dandelion_id\");\n\n  if (!id) {\n    id = generateID();\n    localStorage.setItem(\"dandelion_id\", id);\n  }\n\n  return id;\n}\n\nfunction connectWebsocket(userID: string) {\n  let websocket = new WebSocket(`ws://${window.location.hostname}:${window.location.port}/ws`);\n\n  websocket.onopen = function (evt) {\n    log(\"Websocket: CONNECTED\");\n    websocket.send(`{\"messageType\":\"connect\", \"id\": \"${userID}\"}`);\n\n    let websocketPingInterval = window.setInterval(()=>{websocket.send(`{\"messageType\":\"ping\", \"id\": \"${userID}\"}`);}, 5000)\n  };\n\n  websocket.onclose = function (evt) {\n    log(\"Websocket: DISCONNECTED\");\n  };\n\n  websocket.onmessage = function (evt) {\n    log('Websocket: RESPONSE: ' + evt.data);\n  };\n\n  websocket.onerror = function (evt) {\n    log('Websocket: ERROR: ' + evt);\n  };\n\n  return websocket;\n}\n\nfunction setFont(fontName: string, fontSize: string) {\n\n  let content = document.getElementById('content');\n\n  if (!content) {\n    return;\n  }\n\n  content.style.fontFamily = fontName;\n  content.style.fontSize = fontSize;\n\n  let textArea = document.getElementById('textarea_post');\n  if (!textArea) {\n    return;\n  }\n\n  textArea.style.fontFamily = fontName;\n  textArea.style.fontSize = fontSize;\n}\n\nfunction initOffline() {\n  // Event listener for going offline\n  window.addEventListener('offline', () => { log(\"offline\") });\n\n  // Event listener for going online\n  window.addEventListener('online', () => { log(\"online\") });\n\n  log(`Online status: ${navigator.onLine ? \"online\" : \"offline\"}`)\n\n}\n\nfunction initButtons(userID: string, posts: Post[]) {\n  let font1Button = document.getElementById(\"button_font1\") as HTMLButtonElement;\n  let font2Button = document.getElementById(\"button_font2\") as HTMLButtonElement;\n\n  font1Button.addEventListener('click', () => { setFont('Bookerly', '16px') });\n  font2Button.addEventListener('click', () => { setFont('Virgil', '24px') });\n\n\n  let postButton = document.getElementById(\"button_post\") as HTMLButtonElement;\n  let postText = document.getElementById(\"textarea_post\") as HTMLTextAreaElement;\n\n  if (!(postButton && postText)) {\n    throw new Error();\n  }\n\n  postButton.addEventListener(\"click\", () => {\n    addPost(userID, posts, postText.value);\n    postText.value = \"\";\n  });\n\n}\n\nasync function loadPosts(userID: string) {\n\n  timerStart();\n  let posts: any = await getData(userID, new Date(2022, 8), new Date());\n\n  if (posts.length > 0) {\n    log(`Loaded ${posts.length} posts in ${timerDelta().toFixed(2)}ms`);\n    return posts;\n  }\n\n  posts = await createTestData2(userID);\n\n  log(\"Adding test data...\");\n  addDataArray(userID, posts);\n  return await getData(userID, new Date(2022, 8), new Date());\n}\n\nasync function main() {\n  let posts: Post[] = [];\n  let time = 0; ``\n  let delta = 0;\n\n  let urlParams = (new URL(window.location.href)).searchParams;\n\n  if (urlParams.get(\"sw\") === \"true\") {\n    let registration = await registerServiceWorker();\n  }\n\n  let userID = getUserID();\n  log(`Your user ID is: ${userID}`);\n\n  if (navigator.storage && navigator.storage.persist && !navigator.storage.persisted) {\n    const isPersisted = await navigator.storage.persist();\n    log(`Persisted storage granted: ${isPersisted}`);\n  }\n\n  log(`Persisted: ${(await navigator?.storage?.persisted())?.toString()}`);\n\n  initMarkdown();\n\n  // let main = await fetch(\"/main.js\");\n  // let code = await main.text();\n  //   console.log(code);\n  //   registration.active.postMessage({type:\"updateMain\", code:code});\n\n  posts = await loadPosts(userID);\n\n  let websocket = connectWebsocket(userID);\n  initOffline();\n  initButtons(userID, posts);\n  // debugger;\n\n  timerStart();\n  render(posts);\n  let renderTime = timerDelta();\n\n  log(`render took: ${renderTime.toFixed(2)}ms`);\n\n  log(`memory used: ${((performance as any).memory.usedJSHeapSize / 1024 / 1024).toFixed(2)}Mb`)\n}\n\nfunction render(posts: Post[]) {\n  const fragment = document.createDocumentFragment();\n  let contentDiv = document.getElementById(\"content\");\n  if (!contentDiv) {\n    throw new Error();\n  }\n  contentDiv.innerHTML = \"\";\n  let count = 0;\n\n\n  for (let i = posts.length - 1; i >= 0; i--) {\n    let postData = posts[i];\n\n    let post = renderPost(postData);\n\n    if (post) {\n      fragment.appendChild(post);\n      count++;\n    }\n    if (count > 100) {\n      break;\n    }\n  }\n\n\n  if (!contentDiv) {\n    throw new Error(\"Couldn't get content div!\");\n  }\n\n  contentDiv.appendChild(fragment);\n\n}\n\nfunction renderPost(post: Post) {\n  if (!(post.hasOwnProperty(\"text\"))) {\n    throw new Error(\"Post is malformed!\");\n  }\n  let containerDiv = document.createElement(\"div\");\n  let textDiv = document.createElement(\"div\");\n  let timestampDiv = document.createElement(\"div\");\n  let hr = document.createElement(\"hr\");\n\n\n  textDiv.innerHTML = marked.parse(post.text);\n  // textDiv.innerHTML = DOMPurify.sanitize(marked.parse(post.text));\n\n  timestampDiv.innerText = `${post.post_timestamp.toDateString()}`;\n  timestampDiv.title = `${post.post_timestamp.toLocaleTimeString()} · ${post.post_timestamp.toDateString()}`\n  containerDiv.appendChild(hr);\n  containerDiv.appendChild(textDiv);\n\n  if (!(\"image_data\" in post && post.image_data)) {\n    containerDiv.appendChild(timestampDiv);\n    return containerDiv;\n    // return null;\n  }\n\n  let image = document.createElement(\"img\");\n  const blob = new Blob([post.image_data as ArrayBuffer], { type: 'image/jpg' });\n  const url = URL.createObjectURL(blob);\n  image.onload = () => {\n    URL.revokeObjectURL(url);\n  };\n\n\n  image.src = url;\n  // image.src = image.src = \"data:image/png;base64,\" + post.image;\n  image.className = \"postImage\";\n\n  containerDiv.appendChild(image);\n  containerDiv.appendChild(timestampDiv);\n\n  return containerDiv;\n}\n\nwindow.addEventListener(\"load\", main);\n"]} \ No newline at end of file diff --git a/sw.js b/sw.js index 172cea6..893eac8 100644 --- a/sw.js +++ b/sw.js @@ -51,4 +51,4 @@ addEventListener("message", async (e) => { break; } }); -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3cuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJzcmMvc3cudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLHlCQUF5QjtBQUN6QixNQUFNLFNBQVMsR0FBRyxvQkFBb0IsQ0FBQztBQUV2QyxNQUFNLGNBQWMsR0FBRztJQUN2QixhQUFhO0lBQ2IsVUFBVTtJQUNWLGdCQUFnQjtJQUNoQixRQUFRO0lBQ1IsaUJBQWlCO0lBQ2pCLGVBQWU7SUFDZixjQUFjO0NBQ2IsQ0FBQztBQUVGLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxTQUFTLEVBQUUsQ0FBQyxDQUFLLEVBQUUsRUFBRTtJQUN6QyxDQUFDLENBQUMsU0FBUyxDQUNULENBQUMsS0FBSyxJQUFJLEVBQUU7UUFDVixNQUFNLEtBQUssR0FBRyxNQUFNLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUM7UUFDM0MsT0FBTyxDQUFDLEdBQUcsQ0FDVCxxREFBcUQsRUFDckQsY0FBYyxDQUNmLENBQUM7UUFDRixNQUFNLEtBQUssQ0FBQyxNQUFNLENBQUMsY0FBYyxDQUFDLENBQUM7SUFDckMsQ0FBQyxDQUFDLEVBQUUsQ0FDTCxDQUFDO0FBQ0osQ0FBQyxDQUFDLENBQUM7QUFFSCxJQUFJLENBQUMsZ0JBQWdCLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBSyxFQUFFLEVBQUU7SUFDdkMsQ0FBQyxDQUFDLFdBQVcsQ0FDWCxDQUFDLEtBQUssSUFBSSxFQUFFO1FBQ1YsTUFBTSxDQUFDLEdBQUcsTUFBTSxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUN4QyxJQUFJLENBQUMsRUFBRTtZQUNMLE9BQU8sQ0FBQyxHQUFHLENBQ1QsNENBQTRDLENBQUMsQ0FBQyxPQUFPLENBQUMsR0FBRyxFQUFFLENBQzVELENBQUM7WUFDRixPQUFPLENBQUMsQ0FBQztTQUNWO1FBRUQsSUFBSSxRQUFRLENBQUM7UUFDYixJQUFJO1lBQ0YsT0FBTyxDQUFDLEdBQUcsQ0FDVCw4REFBOEQsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxHQUFHLEVBQUUsQ0FDOUUsQ0FBQztZQUVGLFFBQVEsR0FBRyxNQUFNLEtBQUssQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUM7U0FDbkM7UUFBQyxPQUFPLENBQUMsRUFBRTtZQUNWLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUM7U0FDakI7UUFDRCxNQUFNLEtBQUssR0FBRyxNQUFNLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUM7UUFDM0MsT0FBTyxDQUFDLEdBQUcsQ0FDVCw4Q0FBOEMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxHQUFHLEVBQUUsQ0FDOUQsQ0FBQztRQUVGLElBQUksQ0FBQyxRQUFRLEVBQUU7WUFDYixNQUFNLElBQUksS0FBSyxDQUFDLDZCQUE2QixDQUFDLENBQUMsT0FBTyxDQUFDLEdBQUcsRUFBRSxDQUFDLENBQUE7U0FDOUQ7UUFDRCxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxPQUFPLEVBQUUsUUFBUSxDQUFDLEtBQUssRUFBRSxDQUFDLENBQUM7UUFDdkMsT0FBTyxRQUFRLENBQUM7SUFDbEIsQ0FBQyxDQUFDLEVBQUUsQ0FDTCxDQUFDO0FBQ0osQ0FBQyxDQUFDLENBQUM7QUFFSCxnQkFBZ0IsQ0FBQyxTQUFTLEVBQUUsS0FBSyxFQUFFLENBQUMsRUFBRSxFQUFFO0lBQ3RDLE9BQU8sQ0FBQyxHQUFHLENBQUMscUJBQXFCLENBQUMsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDO0lBRTNDLFFBQVEsQ0FBQyxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUU7UUFDbkIsS0FBSyxZQUFZO1lBQ2YsTUFBTSxLQUFLLEdBQUcsTUFBTSxNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQzNDLE9BQU8sQ0FBQyxHQUFHLENBQUMsZ0RBQWdELENBQUMsQ0FBQztZQUM5RCxLQUFLLENBQUMsR0FBRyxDQUFDLFVBQVUsRUFBRSxJQUFJLFFBQVEsRUFBRSxDQUFDLENBQUM7WUFDdEMsTUFBTTtLQUNUO0FBQ0gsQ0FBQyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBFc3RhYmxpc2ggYSBjYWNoZSBuYW1lXG5jb25zdCBjYWNoZU5hbWUgPSBcImRhbmRlbGlvbl9jYWNoZV92MVwiO1xuXG5jb25zdCBjb250ZW50VG9DYWNoZSA9IFtcblwiL2luZGV4Lmh0bWxcIixcblwiL21haW4uanNcIixcblwiL21hcmtlZC5taW4uanNcIixcblwiL2RiLmpzXCIsXG5cIi9ib29rZXJseS53b2ZmMlwiLFxuXCIvdmlyZ2lsLndvZmYyXCIsXG5cIi9mYXZpY29uLmljb1wiXG5dO1xuXG5zZWxmLmFkZEV2ZW50TGlzdGVuZXIoXCJpbnN0YWxsXCIsIChlOmFueSkgPT4ge1xuICBlLndhaXRVbnRpbChcbiAgICAoYXN5bmMgKCkgPT4ge1xuICAgICAgY29uc3QgY2FjaGUgPSBhd2FpdCBjYWNoZXMub3BlbihjYWNoZU5hbWUpO1xuICAgICAgY29uc29sZS5sb2coXG4gICAgICAgIFwiW1NlcnZpY2UgV29ya2VyXSBDYWNoaW5nIGFsbDogYXBwIHNoZWxsIGFuZCBjb250ZW50XCIsXG4gICAgICAgIGNvbnRlbnRUb0NhY2hlXG4gICAgICApO1xuICAgICAgYXdhaXQgY2FjaGUuYWRkQWxsKGNvbnRlbnRUb0NhY2hlKTtcbiAgICB9KSgpXG4gICk7XG59KTtcblxuc2VsZi5hZGRFdmVudExpc3RlbmVyKFwiZmV0Y2hcIiwgKGU6YW55KSA9PiB7XG4gIGUucmVzcG9uZFdpdGgoXG4gICAgKGFzeW5jICgpID0+IHtcbiAgICAgIGNvbnN0IHIgPSBhd2FpdCBjYWNoZXMubWF0Y2goZS5yZXF1ZXN0KTtcbiAgICAgIGlmIChyKSB7XG4gICAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICAgIGBbU2VydmljZSBXb3JrZXJdIENhY2hlIGhpdCBmb3IgcmVzb3VyY2U6ICR7ZS5yZXF1ZXN0LnVybH1gXG4gICAgICAgICk7XG4gICAgICAgIHJldHVybiByO1xuICAgICAgfVxuXG4gICAgICBsZXQgcmVzcG9uc2U7XG4gICAgICB0cnkge1xuICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICBgW1NlcnZpY2UgV29ya2VyXSBDYWNoZSBtaXNzLCBhdHRlbXB0aW5nIHRvIGZldGNoIHJlc291cmNlOiAke2UucmVxdWVzdC51cmx9YFxuICAgICAgICApO1xuXG4gICAgICAgIHJlc3BvbnNlID0gYXdhaXQgZmV0Y2goZS5yZXF1ZXN0KTtcbiAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgY29uc29sZS53YXJuKGUpO1xuICAgICAgfVxuICAgICAgY29uc3QgY2FjaGUgPSBhd2FpdCBjYWNoZXMub3BlbihjYWNoZU5hbWUpO1xuICAgICAgY29uc29sZS5sb2coXG4gICAgICAgIGBbU2VydmljZSBXb3JrZXJdIEFkZGluZyByZXNvdXJjZSB0byBjYWNoZTogJHtlLnJlcXVlc3QudXJsfWBcbiAgICAgICk7XG5cbiAgICAgIGlmICghcmVzcG9uc2UpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBGYWlsZWQgdG8gZmV0Y2ggcmVzb3VyY2U6ICR7ZS5yZXF1ZXN0LnVybH1gKVxuICAgICAgfVxuICAgICAgY2FjaGUucHV0KGUucmVxdWVzdCwgcmVzcG9uc2UuY2xvbmUoKSk7XG4gICAgICByZXR1cm4gcmVzcG9uc2U7XG4gICAgfSkoKVxuICApO1xufSk7XG5cbmFkZEV2ZW50TGlzdGVuZXIoXCJtZXNzYWdlXCIsIGFzeW5jIChlKSA9PiB7XG4gIGNvbnNvbGUubG9nKGBNZXNzYWdlIHJlY2VpdmVkOiAke2UuZGF0YX1gKTtcblxuICBzd2l0Y2ggKGUuZGF0YS50eXBlKSB7XG4gICAgY2FzZSBcInVwZGF0ZU1haW5cIjpcbiAgICAgIGNvbnN0IGNhY2hlID0gYXdhaXQgY2FjaGVzLm9wZW4oY2FjaGVOYW1lKTtcbiAgICAgIGNvbnNvbGUubG9nKGBbU2VydmljZSBXb3JrZXJdIENhY2hpbmcgbmV3IHJlc291cmNlOiBtYWluLmpzYCk7XG4gICAgICBjYWNoZS5wdXQoXCIvbWFpbi5qc1wiLCBuZXcgUmVzcG9uc2UoKSk7XG4gICAgICBicmVhaztcbiAgfVxufSk7XG4iXX0= \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3cuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJzcmMvc3cudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLHlCQUF5QjtBQUN6QixNQUFNLFNBQVMsR0FBRyxvQkFBb0IsQ0FBQztBQUV2QyxNQUFNLGNBQWMsR0FBRztJQUN2QixhQUFhO0lBQ2IsVUFBVTtJQUNWLGdCQUFnQjtJQUNoQixRQUFRO0lBQ1IsaUJBQWlCO0lBQ2pCLGVBQWU7SUFDZixjQUFjO0NBQ2IsQ0FBQztBQUVGLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxTQUFTLEVBQUUsQ0FBQyxDQUFLLEVBQUUsRUFBRTtJQUN6QyxDQUFDLENBQUMsU0FBUyxDQUNULENBQUMsS0FBSyxJQUFJLEVBQUU7UUFDVixNQUFNLEtBQUssR0FBRyxNQUFNLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUM7UUFDM0MsT0FBTyxDQUFDLEdBQUcsQ0FDVCxxREFBcUQsRUFDckQsY0FBYyxDQUNmLENBQUM7UUFDRixNQUFNLEtBQUssQ0FBQyxNQUFNLENBQUMsY0FBYyxDQUFDLENBQUM7SUFDckMsQ0FBQyxDQUFDLEVBQUUsQ0FDTCxDQUFDO0FBQ0osQ0FBQyxDQUFDLENBQUM7QUFFSCxJQUFJLENBQUMsZ0JBQWdCLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBSyxFQUFFLEVBQUU7SUFDdkMsQ0FBQyxDQUFDLFdBQVcsQ0FDWCxDQUFDLEtBQUssSUFBSSxFQUFFO1FBQ1YsTUFBTSxDQUFDLEdBQUcsTUFBTSxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUN4QyxJQUFJLENBQUMsRUFBRSxDQUFDO1lBQ04sT0FBTyxDQUFDLEdBQUcsQ0FDVCw0Q0FBNEMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxHQUFHLEVBQUUsQ0FDNUQsQ0FBQztZQUNGLE9BQU8sQ0FBQyxDQUFDO1FBQ1gsQ0FBQztRQUVELElBQUksUUFBUSxDQUFDO1FBQ2IsSUFBSSxDQUFDO1lBQ0gsT0FBTyxDQUFDLEdBQUcsQ0FDVCw4REFBOEQsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxHQUFHLEVBQUUsQ0FDOUUsQ0FBQztZQUVGLFFBQVEsR0FBRyxNQUFNLEtBQUssQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUM7UUFDcEMsQ0FBQztRQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7WUFDWCxPQUFPLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2xCLENBQUM7UUFDRCxNQUFNLEtBQUssR0FBRyxNQUFNLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUM7UUFDM0MsT0FBTyxDQUFDLEdBQUcsQ0FDVCw4Q0FBOEMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxHQUFHLEVBQUUsQ0FDOUQsQ0FBQztRQUVGLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQztZQUNkLE1BQU0sSUFBSSxLQUFLLENBQUMsNkJBQTZCLENBQUMsQ0FBQyxPQUFPLENBQUMsR0FBRyxFQUFFLENBQUMsQ0FBQTtRQUMvRCxDQUFDO1FBQ0QsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsT0FBTyxFQUFFLFFBQVEsQ0FBQyxLQUFLLEVBQUUsQ0FBQyxDQUFDO1FBQ3ZDLE9BQU8sUUFBUSxDQUFDO0lBQ2xCLENBQUMsQ0FBQyxFQUFFLENBQ0wsQ0FBQztBQUNKLENBQUMsQ0FBQyxDQUFDO0FBRUgsZ0JBQWdCLENBQUMsU0FBUyxFQUFFLEtBQUssRUFBRSxDQUFDLEVBQUUsRUFBRTtJQUN0QyxPQUFPLENBQUMsR0FBRyxDQUFDLHFCQUFxQixDQUFDLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQztJQUUzQyxRQUFRLENBQUMsQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDcEIsS0FBSyxZQUFZO1lBQ2YsTUFBTSxLQUFLLEdBQUcsTUFBTSxNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQzNDLE9BQU8sQ0FBQyxHQUFHLENBQUMsZ0RBQWdELENBQUMsQ0FBQztZQUM5RCxLQUFLLENBQUMsR0FBRyxDQUFDLFVBQVUsRUFBRSxJQUFJLFFBQVEsRUFBRSxDQUFDLENBQUM7WUFDdEMsTUFBTTtJQUNWLENBQUM7QUFDSCxDQUFDLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIEVzdGFibGlzaCBhIGNhY2hlIG5hbWVcbmNvbnN0IGNhY2hlTmFtZSA9IFwiZGFuZGVsaW9uX2NhY2hlX3YxXCI7XG5cbmNvbnN0IGNvbnRlbnRUb0NhY2hlID0gW1xuXCIvaW5kZXguaHRtbFwiLFxuXCIvbWFpbi5qc1wiLFxuXCIvbWFya2VkLm1pbi5qc1wiLFxuXCIvZGIuanNcIixcblwiL2Jvb2tlcmx5LndvZmYyXCIsXG5cIi92aXJnaWwud29mZjJcIixcblwiL2Zhdmljb24uaWNvXCJcbl07XG5cbnNlbGYuYWRkRXZlbnRMaXN0ZW5lcihcImluc3RhbGxcIiwgKGU6YW55KSA9PiB7XG4gIGUud2FpdFVudGlsKFxuICAgIChhc3luYyAoKSA9PiB7XG4gICAgICBjb25zdCBjYWNoZSA9IGF3YWl0IGNhY2hlcy5vcGVuKGNhY2hlTmFtZSk7XG4gICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgXCJbU2VydmljZSBXb3JrZXJdIENhY2hpbmcgYWxsOiBhcHAgc2hlbGwgYW5kIGNvbnRlbnRcIixcbiAgICAgICAgY29udGVudFRvQ2FjaGVcbiAgICAgICk7XG4gICAgICBhd2FpdCBjYWNoZS5hZGRBbGwoY29udGVudFRvQ2FjaGUpO1xuICAgIH0pKClcbiAgKTtcbn0pO1xuXG5zZWxmLmFkZEV2ZW50TGlzdGVuZXIoXCJmZXRjaFwiLCAoZTphbnkpID0+IHtcbiAgZS5yZXNwb25kV2l0aChcbiAgICAoYXN5bmMgKCkgPT4ge1xuICAgICAgY29uc3QgciA9IGF3YWl0IGNhY2hlcy5tYXRjaChlLnJlcXVlc3QpO1xuICAgICAgaWYgKHIpIHtcbiAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgYFtTZXJ2aWNlIFdvcmtlcl0gQ2FjaGUgaGl0IGZvciByZXNvdXJjZTogJHtlLnJlcXVlc3QudXJsfWBcbiAgICAgICAgKTtcbiAgICAgICAgcmV0dXJuIHI7XG4gICAgICB9XG5cbiAgICAgIGxldCByZXNwb25zZTtcbiAgICAgIHRyeSB7XG4gICAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICAgIGBbU2VydmljZSBXb3JrZXJdIENhY2hlIG1pc3MsIGF0dGVtcHRpbmcgdG8gZmV0Y2ggcmVzb3VyY2U6ICR7ZS5yZXF1ZXN0LnVybH1gXG4gICAgICAgICk7XG5cbiAgICAgICAgcmVzcG9uc2UgPSBhd2FpdCBmZXRjaChlLnJlcXVlc3QpO1xuICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgICBjb25zb2xlLndhcm4oZSk7XG4gICAgICB9XG4gICAgICBjb25zdCBjYWNoZSA9IGF3YWl0IGNhY2hlcy5vcGVuKGNhY2hlTmFtZSk7XG4gICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgYFtTZXJ2aWNlIFdvcmtlcl0gQWRkaW5nIHJlc291cmNlIHRvIGNhY2hlOiAke2UucmVxdWVzdC51cmx9YFxuICAgICAgKTtcblxuICAgICAgaWYgKCFyZXNwb25zZSkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEZhaWxlZCB0byBmZXRjaCByZXNvdXJjZTogJHtlLnJlcXVlc3QudXJsfWApXG4gICAgICB9XG4gICAgICBjYWNoZS5wdXQoZS5yZXF1ZXN0LCByZXNwb25zZS5jbG9uZSgpKTtcbiAgICAgIHJldHVybiByZXNwb25zZTtcbiAgICB9KSgpXG4gICk7XG59KTtcblxuYWRkRXZlbnRMaXN0ZW5lcihcIm1lc3NhZ2VcIiwgYXN5bmMgKGUpID0+IHtcbiAgY29uc29sZS5sb2coYE1lc3NhZ2UgcmVjZWl2ZWQ6ICR7ZS5kYXRhfWApO1xuXG4gIHN3aXRjaCAoZS5kYXRhLnR5cGUpIHtcbiAgICBjYXNlIFwidXBkYXRlTWFpblwiOlxuICAgICAgY29uc3QgY2FjaGUgPSBhd2FpdCBjYWNoZXMub3BlbihjYWNoZU5hbWUpO1xuICAgICAgY29uc29sZS5sb2coYFtTZXJ2aWNlIFdvcmtlcl0gQ2FjaGluZyBuZXcgcmVzb3VyY2U6IG1haW4uanNgKTtcbiAgICAgIGNhY2hlLnB1dChcIi9tYWluLmpzXCIsIG5ldyBSZXNwb25zZSgpKTtcbiAgICAgIGJyZWFrO1xuICB9XG59KTtcbiJdfQ== \ No newline at end of file