diff --git a/CLAUDE.md b/CLAUDE.md index c65ea81..3edbf8a 100644 --- a/CLAUDE.md +++ b/CLAUDE.md @@ -130,6 +130,21 @@ if timeout > DefaultTimeoutSeconds { - Provide public API methods (`IsEnabled()`, `CheckPolicy()`) - Never change unexported→exported to fix bugs +### 6. Auth-Required Configuration (CAUTION) + +**Be extremely careful when modifying auth-related settings in deployment configs.** + +The `ORLY_AUTH_REQUIRED` and `ORLY_AUTH_TO_WRITE` settings control whether clients must authenticate via NIP-42 before interacting with the relay. Changing these on a production relay can: + +- **Lock out all existing clients** if they don't support NIP-42 auth +- **Break automated systems** (bots, bridges, scrapers) that depend on anonymous access +- **Cause data sync issues** if upstream relays can't push events + +Before enabling auth-required on any deployment: +1. Verify all expected clients support NIP-42 +2. Ensure the relay identity key is properly configured +3. Test with a non-production instance first + ## Database Backends | Backend | Use Case | Build | diff --git a/app/web/dist/bundle.js b/app/web/dist/bundle.js index 309d515..d63fb96 100644 --- a/app/web/dist/bundle.js +++ b/app/web/dist/bundle.js @@ -16,7 +16,7 @@ const It=BigInt("0xfffffffffffffffffffffffffffffffffffffffffffffffffffffffefffff Object.defineProperty(e,"__esModule",{value:!0}),e.bytes=e.stringToBytes=e.str=e.bytesToString=e.hex=e.utf8=e.bech32m=e.bech32=e.base58check=e.base58xmr=e.base58xrp=e.base58flickr=e.base58=e.base64url=e.base64=e.base32crockford=e.base32hex=e.base32=e.base16=e.utils=e.assertNumber=void 0,e.assertNumber=t;const a=(e,t)=>t?a(t,e%t):e,c=(e,t)=>e+(t-a(e,t));function u(e,n,i,r){if(!Array.isArray(e))throw new Error("convertRadix2: data should be array");if(n<=0||n>32)throw new Error(`convertRadix2: wrong from=${n}`);if(i<=0||i>32)throw new Error(`convertRadix2: wrong to=${i}`);if(c(n,i)>32)throw new Error(`convertRadix2: carry overflow from=${n} to=${i} carryBits=${c(n,i)}`);let s=0,o=0;const l=2**i-1,a=[];for(const r of e){if(t(r),r>=2**n)throw new Error(`convertRadix2: invalid data word=${r} from=${n}`);if(s=s<32)throw new Error(`convertRadix2: carry overflow pos=${o} from=${n}`);for(o+=n;o>=i;o-=i)a.push((s>>o-i&l)>>>0);s&=2**o-1}if(s=s<=n)throw new Error("Excess padding");if(!r&&s)throw new Error(`Non-zero padding: ${s}`);return r&&o>0&&a.push(s>>>0),a}function d(e){return t(e),{encode:t=>{if(!(t instanceof Uint8Array))throw new Error("radix.encode input should be Uint8Array");return l(Array.from(t),256,e)},decode:t=>{if(!Array.isArray(t)||t.length&&"number"!=typeof t[0])throw new Error("radix.decode input should be array of strings");return Uint8Array.from(l(t,e,256))}}}function f(e,n=!1){if(t(e),e<=0||e>32)throw new Error("radix2: bits should be in (0..32]");if(c(8,e)>32||c(e,8)>32)throw new Error("radix2: carry overflow");return{encode:t=>{if(!(t instanceof Uint8Array))throw new Error("radix2.encode input should be Uint8Array");return u(Array.from(t),8,e,!n)},decode:t=>{if(!Array.isArray(t)||t.length&&"number"!=typeof t[0])throw new Error("radix2.decode input should be array of strings");return Uint8Array.from(u(t,e,8,n))}}}function p(e){if("function"!=typeof e)throw new Error("unsafeWrapper fn should be function");return function(...t){try{return e.apply(null,t)}catch(e){}}}function h(e,n){if(t(e),"function"!=typeof n)throw new Error("checksum fn should be function");return{encode(t){if(!(t instanceof Uint8Array))throw new Error("checksum.encode: input should be Uint8Array");const i=n(t).slice(0,e),r=new Uint8Array(t.length+e);return r.set(t),r.set(i,t.length),r},decode(t){if(!(t instanceof Uint8Array))throw new Error("checksum.decode: input should be Uint8Array");const i=t.slice(0,-e),r=n(i).slice(0,e),s=t.slice(-e);for(let t=0;te.toUpperCase().replace(/O/g,"0").replace(/[IL]/g,"1"))),e.base64=n(f(6),i("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"),s(6),r("")),e.base64url=n(f(6),i("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_"),s(6),r(""));const g=e=>n(d(58),i(e),r(""));e.base58=g("123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz"),e.base58flickr=g("123456789abcdefghijkmnopqrstuvwxyzABCDEFGHJKLMNPQRSTUVWXYZ"),e.base58xrp=g("rpshnaf39wBUDNEGHJKLM4PQRST7VWXYZ2bcdeCg65jkm8oFqi1tuvAxyz");const m=[0,2,3,5,6,7,9,10,11];e.base58xmr={encode(t){let n="";for(let i=0;in(h(4,e=>t(t(e))),e.base58);const y=n(i("qpzry9x8gf2tvdw0s3jn54khce6mua7l"),r("")),v=[996825010,642813549,513874426,1027748829,705979059];function w(e){const t=e>>25;let n=(33554431&e)<<5;for(let e=0;e>e&1)&&(n^=v[e]);return n}function A(e,t,n=1){const i=e.length;let r=1;for(let t=0;t126)throw new Error(`Invalid prefix (${e})`);r=w(r)^n>>5}r=w(r);for(let t=0;tn)throw new TypeError(`Wrong string length: ${e.length} (${e}). Expected (8..${n})`);const i=e.toLowerCase();if(e!==i&&e!==e.toUpperCase())throw new Error("String must be lowercase or uppercase");const r=(e=i).lastIndexOf("1");if(0===r||-1===r)throw new Error('Letter "1" must be present between prefix and data only');const s=e.slice(0,r),o=e.slice(r+1);if(o.length<6)throw new Error("Data must be at least 6 characters long");const l=y.decode(o).slice(0,-6),a=A(s,l,t);if(!o.endsWith(a))throw new Error(`Invalid checksum in ${e}: expected "${a}"`);return{prefix:s,words:l}}return{encode:function(e,n,i=90){if("string"!=typeof e)throw new Error("bech32.encode prefix should be string, not "+typeof e);if(!Array.isArray(n)||n.length&&"number"!=typeof n[0])throw new Error("bech32.encode words should be array of numbers, not "+typeof n);const r=e.length+7+n.length;if(!1!==i&&r>i)throw new TypeError(`Length ${r} exceeds limit ${i}`);return`${e=e.toLowerCase()}1${y.encode(n)}${A(e,n,t)}`},decode:o,decodeToBytes:function(e){const{prefix:t,words:n}=o(e,!1);return{prefix:t,words:n,bytes:i(n)}},decodeUnsafe:p(o),fromWords:i,fromWordsUnsafe:s,toWords:r}}e.bech32=b("bech32"),e.bech32m=b("bech32m"),e.utf8={encode:e=>(new TextDecoder).decode(e),decode:e=>(new TextEncoder).encode(e)},e.hex=n(f(4),i("0123456789abcdef"),r(""),o(e=>{if("string"!=typeof e||e.length%2)throw new TypeError(`hex.decode: expected string, got ${typeof e} with length ${e.length}`);return e.toLowerCase()}));const k={utf8:e.utf8,hex:e.hex,base16:e.base16,base32:e.base32,base64:e.base64,base64url:e.base64url,base58:e.base58,base58xmr:e.base58xmr},I=`Invalid encoding type. Available types: ${Object.keys(k).join(", ")}`;e.bytesToString=(e,t)=>{if("string"!=typeof e||!k.hasOwnProperty(e))throw new TypeError(I);if(!(t instanceof Uint8Array))throw new TypeError("bytesToString() expects Uint8Array");return k[e].encode(t)},e.str=e.bytesToString;e.stringToBytes=(e,t)=>{if(!k.hasOwnProperty(e))throw new TypeError(I);if("string"!=typeof t)throw new TypeError("stringToBytes() expects string");return k[e].decode(t)},e.bytes=e.stringToBytes}({}),BigInt(1e3),BigInt(1e6),BigInt(1e9),BigInt(1e12),BigInt("2100000000000000000"),BigInt(1e11);const Sd={payment_hash:1,payment_secret:16,description:13,payee:19,description_hash:23,expiry:6,min_final_cltv_expiry:24,fallback_address:9,route_hint:3,feature_bits:5,metadata:27};for(let e=0,t=Object.keys(Sd);eJc(e.tags,e=>zc(e)?e:void 0,zu))}(e),function(e){if(function(e){return nd(e)&&Reflect.has(e,Qd)}(e))return e[Qd];const t=id(e);if(!t)return;const n=Jc(t,e=>zc(e)?e:void 0,zu);return Reflect.set(e,Qd,n),n}(e))}gd.extend("EncryptedContentCache"),gd.extend("event-cache"),new TextDecoder;const Dd=Symbol.for("mailboxes-inboxes"),$d=Symbol.for("mailboxes-outboxes");function Pd(e){return eu(e,Dd,()=>{const t=[];for(const n of e.tags)if(jc(n))try{const[,e,i]=n;!e||!Eu(e)||t.includes(e)||"read"!==i&&void 0!==i||t.push(bu(e))}catch{}return t})}function Rd(e){return eu(e,$d,()=>{const t=[];for(const n of e.tags)if(jc(n))try{const[e,i,r]=n;"r"!==e||!Eu(i)||t.includes(i)||"write"!==r&&void 0!==r||t.push(bu(i))}catch{}return t})}const Td=Symbol.for("mute-public"),Ud=Symbol.for("mute-hidden");function _d(e){return{pubkeys:new Set(e.filter(zc).map(e=>e[1])),threads:new Set(e.filter(Mc).map(e=>e[1])),hashtags:new Set(e.filter(Gc).map(e=>e[1].toLocaleLowerCase())),words:new Set(e.filter(e=>"word"===e[0]&&e[1]).map(e=>e[1].toLocaleLowerCase()))}}function Nd(e){const t=function(e){if(function(e){return nd(e)&&Reflect.has(e,Ud)}(e))return e[Ud];const t=id(e);if(!t)return;const n=_d(t);return Reflect.set(e,Ud,n),n}(e),n=function(e){return eu(e,Td,()=>_d(e.tags))}(e);return t?function(...e){const t={pubkeys:new Set,threads:new Set,hashtags:new Set,words:new Set};for(const n of e){for(const e of n.pubkeys)t.pubkeys.add(e);for(const e of n.threads)t.threads.add(e);for(const e of n.hashtags)t.hashtags.add(e);for(const e of n.words)t.words.add(e)}return t}(t,n):n}var Ld;!function(e){e.nudity="nudity",e.malware="malware",e.profanity="profanity",e.illegal="illegal",e.spam="spam",e.impersonation="impersonation",e.other="other"}(Ld||(Ld={}));const Od=Symbol.for("nip10-thread-refs");function Md(e){if(!e[1])throw new Error("Missing event id in tag");let t={id:e[1]};return e[2]&&Eu(e[2])&&(t.relays=[e[2]]),"e"!==e[0]||"root"!==e[3]&&"reply"!==e[3]&&"mention"!==e[3]||!e[4]||64!==e[4].length||(t.author=e[4]),t}function zd(e){return t=>{const n=new Set;return t.pipe(Lc(t=>{if(void 0!==t)if(Array.isArray(t))for(const i of t)n.has(i)||(n.add(i),e.claim(i));else n.has(t)||(n.add(t),e.claim(t))}),Pc(()=>{for(const t of n)e.removeClaim(t)}))}}function jd(e){return t=>{let n;return t.pipe(Lc(t=>{n!==t&&(n&&e.removeClaim(n),t&&e.claim(t),n=t)}),Pc(()=>{n&&e.removeClaim(n)}))}}function Hd(){return e=>e.pipe(Bc(e=>null!=e))}function Gd(e){return t=>new Sa(n=>{let i=!1;const r=t.subscribe({next:e=>{i=!0,n.next(e)},error:e=>n.error(e),complete:()=>n.complete()});return i||n.next(e),r})}function Jd(e){return"string"==typeof e&&(e={id:e}),t=>Sc(Ec(()=>function(e,t){const n=e.getEvent(t.id);return n instanceof Promise?uc(n):dc(n)}(t,e)).pipe(function(e,t){return Nc(n=>n?dc(n):e.eventLoader?uc(e.eventLoader(t)):ja)}(t,e),Hd()),t.insert$.pipe(Bc(t=>t.id===e.id)),t.remove$.pipe(Bc(t=>t.id===e.id),Qc(1),Qa(function(e,t){e.subscribe(Fa(t,ma))}),$c(void 0))).pipe(jd(t),Fc((e,t)=>e?.id===t?.id),Gd(void 0))}function Kd(e){return t=>{let n;return Sc(Ec(()=>function(e,t){const n=e.getReplaceable(t.kind,t.pubkey,t.identifier);return n instanceof Promise?uc(n):dc(n)}(t,e)).pipe(function(e,t){return Nc(n=>n?dc(n):void 0!==t.identifier?e.addressableLoader?uc(e.addressableLoader(t)).pipe(Bc(e=>!!e)):ja:e.replaceableLoader?uc(e.replaceableLoader(t)).pipe(Bc(e=>!!e)):ja)}(t,e),Hd()),t.insert$.pipe(Bc(t=>t.pubkey==e.pubkey&&t.kind===e.kind&&(void 0===e.identifier||fu(t)===e.identifier)))).pipe(Fc((e,t)=>e.created_at>=t.created_at),Lc(e=>n=e),(i=t.remove$.pipe(Bc(e=>e.id===n?.id)),Qa(function(e,t){ic(i).subscribe(Fa(t,function(){return t.complete()},ma)),!t.closed&&e.subscribe(t)})),$c(void 0),function(e){var t,n,i=1/0;return null!=e&&("object"==typeof e?(t=e.count,i=void 0===t?1/0:t,n=e.delay):i=e),i<=0?function(){return ja}:Qa(function(e,t){var r,s=0,o=function(){if(null==r||r.unsubscribe(),r=null,null!=n){var e="number"==typeof n?xc(n):ic(n(s)),i=Fa(t,function(){i.unsubscribe(),l()});e.subscribe(i)}else l()},l=function(){var n=!1;r=e.subscribe(Fa(t,void 0,function(){++s{const i=new Map;return Ec(()=>{const t=n.getTimeline(e);return t instanceof Promise?uc(t):dc(t)}).pipe(zd(n),Tc(n.insert$.pipe(Bc(t=>kd(e,t)),zd(n))),Tc(n.remove$.pipe(Bc(t=>kd(e,t)),fc(e=>e.id))),Uc((e,n)=>{if("string"==typeof n)return e.filter(e=>e.id!==n);if(Array.isArray(n)){if(!t)for(const e of n)au(e.kind)&&i.set(cu(e),e);return n}let r=[...e];if(!t&&au(n.kind)){const t=cu(n),s=i.get(t);if(s&&n.created_ati.clear()))}}function qd(e){return t=>Ac(Object.fromEntries(e.map(e=>[e,t.model(Jd,{id:e})])))}function Yd(e){return t=>Ac(Object.fromEntries(e.map(e=>[du(e.kind,e.pubkey,e.identifier),t.model(Kd,e)])))}function Wd(e){return t=>{let n;return Sc(t.pipe(Lc(e=>n=e)),e.update$.pipe(Bc(e=>e.id===n?.id)))}}function Zd(e){return"string"==typeof e&&(e={pubkey:e}),t=>t.replaceable({kind:pr.Contacts,pubkey:e.pubkey,relays:e.relays}).pipe(Wd(t),fc(e=>e?Fd(e):[]))}function Xd(e){return t=>{const n=[{kinds:[1111],"#e":[e.id]}];return Vl(e.kind)&&n.push({kinds:[1111],"#a":[uu(e)]}),t.timeline(n)}}function ef(e){return"string"==typeof e&&(e={pubkey:e}),t=>t.replaceable({kind:pr.RelayList,pubkey:e.pubkey,relays:e.relays}).pipe(fc(e=>e&&{inboxes:Pd(e),outboxes:Rd(e)}))}function tf(e){return"string"==typeof e&&(e={pubkey:e}),t=>t.replaceable({kind:pr.Mutelist,pubkey:e.pubkey,relays:e.relays}).pipe(Wd(t),fc(e=>e&&Nd(e)))}function nf(e){return"string"==typeof e&&(e={pubkey:e}),t=>t.replaceable({kind:pr.Metadata,pubkey:e.pubkey,relays:e.relays}).pipe(Bc(xd),fc(e=>e&&Ed(e)),Gd(void 0))}function rf(e){return t=>t.timeline(au(e.kind)?[{kinds:[pr.Reaction],"#e":[e.id]},{kinds:[pr.Reaction],"#a":[cu(e)]}]:[{kinds:[pr.Reaction],"#e":[e.id]}])}const sf={kinds:[pr.ShortTextNote]};function of(e,t){const n=new Map,i=new Map,{kinds:r}={...sf,...t};let s="";const o={},l={kinds:r};var a;return"string"!=typeof(a=e)&&Reflect.has(a,"identifier")&&Reflect.has(a,"pubkey")&&Reflect.has(a,"kind")?(s=ju(e),o.kinds=[e.kind],o.authors=[e.pubkey],o["#d"]=[e.identifier],l["#a"]=[s]):"string"==typeof e?(s=e,o.ids=[e],l["#e"]=[e]):(s=e.id,o.ids=[e.id],l["#e"]=[e.id]),e=>e.filters([o,l]).pipe(fc(e=>{if(!i.has(cu(e))){const t=function(e){return eu(e,Od,()=>{const t=function(e){const t=e.filter(e=>"e"===e[0]&&e[1]),n=e.filter(e=>"a"===e[0]&&e[1]);let i=t.find(e=>"root"===e[3]),r=t.find(e=>"reply"===e[3]),s=n.find(e=>"root"===e[3]),o=n.find(e=>"reply"===e[3]);if(i&&r||(i=r=i||r),s&&o||(s=o=s||o),!i&&!r){const e=t.filter(e=>!e[3]);e.length>=1&&(i=e[0],r=e[e.length-1]??i)}return{root:i||s?{e:i,a:s}:void 0,reply:r||o?{e:r,a:o}:void 0}}(e.tags);let n,i;if(t.root)try{n={e:t.root.e&&Md(t.root.e),a:t.root.a&&Mu(t.root.a)}}catch(e){}if(t.reply)try{i={e:t.reply.e&&Md(t.reply.e),a:t.reply.a&&Mu(t.reply.a)}}catch(e){}return{root:n,reply:i}})}(e),r=n.get(cu(e))||new Set,s={event:e,refs:t,replies:r};for(const e of r)e.parent=s;if(t.reply?.e||t.reply?.a){let e=t.reply.e?t.reply.e.id:ju(t.reply.a);if(s.parent=i.get(e),s.parent)s.parent.replies.add(s);else{let t=n.get(e);t||(t=new Set,n.set(e,t)),t.add(s)}}i.set(cu(e),s)}return{root:i.get(s),all:i}}))}function lf(e){return class extends e{models=new Map;modelKeepWarm=6e4;model(e,...t){let n=this.models.get(e);n||(n=new Map,this.models.set(e,n));const i=e.getKey?e.getKey(...t):Ad(t);let r=n.get(i);if(!r){const s=()=>{n.get(i)===r&&n.delete(i)};r=e(...t)(this).pipe(Pc(s),function(e){void 0===e&&(e={});var t=e.connector,n=void 0===t?function(){return new Pa}:t,i=e.resetOnError,r=void 0===i||i,s=e.resetOnComplete,o=void 0===s||s,l=e.resetOnRefCountZero,a=void 0===l||l;return function(e){var t,i,s,l=0,c=!1,u=!1,d=function(){null==i||i.unsubscribe(),i=void 0},f=function(){d(),t=s=void 0,c=u=!1},p=function(){var e=t;f(),null==e||e.unsubscribe()};return Qa(function(e,h){l++,u||c||d();var g=s=null!=s?s:n();h.add(function(){0!==--l||u||c||(i=_c(p,a))}),g.subscribe(h),!t&&l>0&&(t=new ka({next:function(e){return g.next(e)},error:function(e){u=!0,d(),i=_c(f,r,e),g.error(e)},complete:function(){c=!0,d(),i=_c(f,o),g.complete()}}),ic(e).subscribe(t))})(e)}}({connector:()=>new Ua(1),resetOnComplete:()=>xc(this.modelKeepWarm),resetOnRefCountZero:()=>xc(this.modelKeepWarm)})),n.set(i,r)}return r}filters(e,t=!1){e=Array.isArray(e)?e:[e];const n=this.getByFilters(e);return Sc(t?ja:n&&"function"==typeof n.then?uc(n).pipe(kc(e=>uc(Array.from(e)))):uc(Array.from(n)),this.insert$.pipe(Bc(t=>kd(e,t))))}event(e){return"string"==typeof e&&(e={id:e}),this.model(Jd,e)}replaceable(...e){let t;if(1===e.length)t=e[0];else if(3===e.length||2===e.length){let[n,i,r]=e;t={kind:n,pubkey:i,identifier:r}}if(!t)throw new Error("Invalid arguments, expected address pointer or kind, pubkey, identifier");return this.model(Kd,t)}addressable(e){return this.model(Kd,e)}timeline(e,t=!1){return this.model(Vd,e,t)}profile(e){return this.model(nf,e)}contacts(e){return"string"==typeof e&&(e={pubkey:e}),this.model(Zd,e)}mutes(e){return"string"==typeof e&&(e={pubkey:e}),this.model(tf,e)}mailboxes(e){return"string"==typeof e&&(e={pubkey:e}),this.model(ef,e)}blossomServers(e){return"string"==typeof e&&(e={pubkey:e}),this.model(Id,e)}reactions(e){return this.model(rf,e)}thread(e){return this.model(of,e)}comments(e){return this.model(Xd,e)}events(e){return this.model(qd,e)}replaceableSet(e){return this.model(Yd,e)}}}class af extends(lf(class{})){database;memory;keepOldVersions=!1;keepExpired=!1;verifyEvent;insert$=new Pa;update$=new Pa;remove$=new Pa;eventLoader;replaceableLoader;addressableLoader;constructor(e=new md){super(),e?(this.database=e,this.memory=new md):this.database=this.memory=new md,this.insert$.subscribe(e=>{Reflect.set(e,iu,this)}),this.remove$.subscribe(e=>{Reflect.deleteProperty(e,iu)})}mapToMemory(e){if(void 0!==e)return this.memory?this.memory.add(e):e}deletedIds=new Set;deletedCoords=new Map;checkDeleted(e){if("string"==typeof e)return this.deletedIds.has(e);if(this.deletedIds.has(e.id))return!0;if(Vl(e.kind)){const t=e.tags.find(e=>"d"===e[0])?.[1],n=this.deletedCoords.get(du(e.kind,e.pubkey,t));if(n)return n>e.created_at}return!1}expirations=new Map;addExpiration(e){const t=gu(e);t&&Number.isFinite(t)&&this.expirations.set(e.id,t)}expirationTimeout=null;nextExpirationCheck=null;handleExpiringEvent(e){const t=gu(e);if(!t)return;if(this.expirations.set(e.id,t),this.expirationTimeout&&this.nextExpirationCheck&&this.nextExpirationChecke[1])}(e);for(const e of t)this.deletedIds.add(e),this.remove(e);const n=function(e){return e.tags.filter(Hc).map(e=>e[1])}(e);for(const t of n){this.deletedCoords.set(t,Math.max(this.deletedCoords.get(t)??0,e.created_at));const n=Ou(t);if(!n)continue;const i=this.database.getReplaceableHistory(n.kind,n.pubkey,n.identifier)??[];for(const t of i)t.created_at"d"===e[0])?.[1]:void 0;if(!this.keepOldVersions&&au(e.kind)){const t=this.database.getReplaceableHistory(e.kind,e.pubkey,i);if(t&&t.length>0&&t[0].created_at>=e.created_at)return af.mergeDuplicateEvent(e,t[0]),t[0]}if(this.verifyEvent&&!1===this.verifyEvent(e))return null;const r=this.memory?.add(e);if(r&&r!==e)return af.mergeDuplicateEvent(e,r),t&&Iu(r,t),r;const s=this.mapToMemory(this.database.add(e));if(e!==s&&af.mergeDuplicateEvent(e,s),t&&Iu(s,t),s===e&&this.insert$.next(s),!this.keepOldVersions&&au(e.kind)){const t=this.database.getReplaceableHistory(e.kind,e.pubkey,i);if(t&&t.length>0){const n=Array.from(t).filter(t=>t.created_atthis.mapToMemory(e)??e)}getByFilters(e){const t=this.database.getByFilters(e);return this.memory?t.map(e=>this.mapToMemory(e)):t}getTimeline(e){const t=this.database.getTimeline(e);return this.memory?t.map(e=>this.mapToMemory(e)):t}touch(e){return this.memory?.touch(e)}claim(e){return this.memory?.claim(e)}isClaimed(e){return this.memory?.isClaimed(e)??!1}removeClaim(e){return this.memory?.removeClaim(e)}clearClaim(e){return this.memory?.clearClaim(e)}unclaimed(){return this.memory?.unclaimed()||function*(){}()}prune(e){return this.memory?.prune(e)??0}removed(e){return this.checkDeleted(e)?ja:this.remove$.pipe(Bc(t=>t.id===e),Qc(1),kc(()=>ja))}updated(e){return this.update$.pipe(Bc(t=>t.id===e||t===e))}}function cf(){let e,t;const n=new Promise((n,i)=>{e=n,t=i});return n.resolve=e,n.reject=t,n}"navigator"in globalThis&&navigator.userAgent.includes("Android")&&navigator.clipboard&&navigator.clipboard.readText;var uf,df;qu(pr.NostrConnect,"nip44"),function(e){e.GetPublicKey="get_pubic_key",e.SignEvent="sign_event",e.Nip04Encrypt="nip04_encrypt",e.Nip04Decrypt="nip04_decrypt",e.Nip44Encrypt="nip44_encrypt",e.Nip44Decrypt="nip44_decrypt"}(uf||(uf={})),function(e){e.Connect="connect",e.CreateAccount="create_account",e.GetPublicKey="get_public_key",e.SignEvent="sign_event",e.Nip04Encrypt="nip04_encrypt",e.Nip04Decrypt="nip04_decrypt",e.Nip44Encrypt="nip44_encrypt",e.Nip44Decrypt="nip44_decrypt"}(df||(df={}));class ff{key;constructor(e){this.key=e||cr()}async getPublicKey(){return ur(this.key)}async signEvent(e){return dr(e,this.key)}nip04={encrypt:async(e,t)=>po.encrypt(this.key,e,t),decrypt:async(e,t)=>po.decrypt(this.key,e,t)};nip44={encrypt:async(e,t)=>Fo.v2.encrypt(t,Fo.v2.utils.getConversationKey(this.key,e)),decrypt:async(e,t)=>Fo.v2.decrypt(t,Fo.v2.utils.getConversationKey(this.key,e))};static fromKey(e){return new ff(function(e){if(e instanceof Uint8Array)return e;if(xu(e))return Lu(e);{const t=Ys.decode(e);if("nsec"!==t.type)throw new Error(`Cant get secret key from ${t.type}`);return t.data}}(e))}} /*! noble-secp256k1 - MIT License (c) 2019 Paul Miller (paulmillr.com) */const pf=BigInt(0),hf=BigInt(1),gf=BigInt(2),mf=BigInt(3),yf=BigInt(8),vf=Object.freeze({a:pf,b:BigInt(7),P:BigInt("0xfffffffffffffffffffffffffffffffffffffffffffffffffffffffefffffc2f"),n:BigInt("0xfffffffffffffffffffffffffffffffebaaedce6af48a03bbfd25e8cd0364141"),h:hf,Gx:BigInt("55066263022277343669578718895168534326250603453777594175500187360389116729240"),Gy:BigInt("32670510020758816978083085130507043184471273380659243275938904335757337482424"),beta:BigInt("0x7ae96a2b657c07106e64479eac3434e99cf0497512f58995c1396c28719501ee")}),wf=(e,t)=>(e+t/gf)/t,Af={beta:BigInt("0x7ae96a2b657c07106e64479eac3434e99cf0497512f58995c1396c28719501ee"),splitScalar(e){const{n:t}=vf,n=BigInt("0x3086d221a7d46bcde86c90e49284eb15"),i=-hf*BigInt("0xe4437ed6010e88286f547fa90abfe4c3"),r=BigInt("0x114ca50f7a8e2f3f657c1108d9d44cfd8"),s=n,o=BigInt("0x100000000000000000000000000000000"),l=wf(s*e,t),a=wf(-i*e,t);let c=Gf(e-l*n-a*r,t),u=Gf(-l*i-a*s,t);const d=c>o,f=u>o;if(d&&(c=t-c),f&&(u=t-u),c>o||u>o)throw new Error("splitScalarEndo: Endomorphism failed, k="+e);return{k1neg:d,k1:c,k2neg:f,k2:u}}},bf=32;function kf(e){const{a:t,b:n}=vf,i=Gf(e*e),r=Gf(i*e);return Gf(r+t*e+n)}const If=vf.a===pf;function Cf(e){if(!(e instanceof Ef))throw new TypeError("JacobianPoint expected")}class Ef{constructor(e,t,n){this.x=e,this.y=t,this.z=n}static fromAffine(e){if(!(e instanceof Bf))throw new TypeError("JacobianPoint#fromAffine: expected Point");return e.equals(Bf.ZERO)?Ef.ZERO:new Ef(e.x,e.y,hf)}static toAffineBatch(e){const t=function(e,t=vf.P){const n=new Array(e.length),i=e.reduce((e,i,r)=>i===pf?e:(n[r]=e,Gf(e*i,t)),hf),r=Kf(i,t);return e.reduceRight((e,i,r)=>i===pf?e:(n[r]=Gf(e*n[r],t),Gf(e*i,t)),r),n}(e.map(e=>e.z));return e.map((e,n)=>e.toAffine(t[n]))}static normalizeZ(e){return Ef.toAffineBatch(e).map(Ef.fromAffine)}equals(e){Cf(e);const{x:t,y:n,z:i}=this,{x:r,y:s,z:o}=e,l=Gf(i*i),a=Gf(o*o),c=Gf(t*a),u=Gf(r*l),d=Gf(Gf(n*o)*a),f=Gf(Gf(s*i)*l);return c===u&&d===f}negate(){return new Ef(this.x,Gf(-this.y),this.z)}double(){const{x:e,y:t,z:n}=this,i=Gf(e*e),r=Gf(t*t),s=Gf(r*r),o=e+r,l=Gf(gf*(Gf(o*o)-i-s)),a=Gf(mf*i),c=Gf(a*a),u=Gf(c-gf*l),d=Gf(a*(l-u)-yf*s),f=Gf(gf*t*n);return new Ef(u,d,f)}add(e){Cf(e);const{x:t,y:n,z:i}=this,{x:r,y:s,z:o}=e;if(r===pf||s===pf)return this;if(t===pf||n===pf)return e;const l=Gf(i*i),a=Gf(o*o),c=Gf(t*a),u=Gf(r*l),d=Gf(Gf(n*o)*a),f=Gf(Gf(s*i)*l),p=Gf(u-c),h=Gf(f-d);if(p===pf)return h===pf?this.double():Ef.ZERO;const g=Gf(p*p),m=Gf(p*g),y=Gf(c*g),v=Gf(h*h-m-gf*y),w=Gf(h*(y-v)-d*m),A=Gf(i*o*p);return new Ef(v,w,A)}subtract(e){return this.add(e.negate())}multiplyUnsafe(e){const t=Ef.ZERO;if("bigint"==typeof e&&e===pf)return t;let n=Hf(e);if(n===hf)return this;if(!If){let e=t,i=this;for(;n>pf;)n&hf&&(e=e.add(i)),i=i.double(),n>>=hf;return e}let{k1neg:i,k1:r,k2neg:s,k2:o}=Af.splitScalar(n),l=t,a=t,c=this;for(;r>pf||o>pf;)r&hf&&(l=l.add(c)),o&hf&&(a=a.add(c)),c=c.double(),r>>=hf,o>>=hf;return i&&(l=l.negate()),s&&(a=a.negate()),a=new Ef(Gf(a.x*Af.beta),a.y,a.z),l.add(a)}precomputeWindow(e){const t=If?128/e+1:256/e+1,n=[];let i=this,r=i;for(let s=0;s>=u,o>l&&(o-=c,e+=hf);const d=n,f=n+Math.abs(o)-1,p=t%2!=0,h=o<0;0===o?s=s.add(xf(p,i[d])):r=r.add(xf(h,i[f]))}return{p:r,f:s}}multiply(e,t){let n,i,r=Hf(e);if(If){const{k1neg:e,k1:s,k2neg:o,k2:l}=Af.splitScalar(r);let{p:a,f:c}=this.wNAF(s,t),{p:u,f:d}=this.wNAF(l,t);a=xf(e,a),u=xf(o,u),u=new Ef(Gf(u.x*Af.beta),u.y,u.z),n=a.add(u),i=c.add(d)}else{const{p:e,f:s}=this.wNAF(r,t);n=e,i=s}return Ef.normalizeZ([n,i])[0]}toAffine(e){const{x:t,y:n,z:i}=this,r=this.equals(Ef.ZERO);null==e&&(e=r?yf:Kf(i));const s=e,o=Gf(s*s),l=Gf(o*s),a=Gf(t*o),c=Gf(n*l),u=Gf(i*s);if(r)return Bf.ZERO;if(u!==hf)throw new Error("invZ was invalid");return new Bf(a,c)}}function xf(e,t){const n=t.negate();return e?n:t}Ef.BASE=new Ef(vf.Gx,vf.Gy,hf),Ef.ZERO=new Ef(pf,hf,pf);const Sf=new WeakMap;class Bf{constructor(e,t){this.x=e,this.y=t}_setWindowSize(e){this._WINDOW_SIZE=e,Sf.delete(this)}hasEvenY(){return this.y%gf===pf}static fromCompressedHex(e){const t=32===e.length,n=zf(t?e:e.subarray(1));if(!qf(n))throw new Error("Point is not on curve");let i=function(e){const{P:t}=vf,n=BigInt(6),i=BigInt(11),r=BigInt(22),s=BigInt(23),o=BigInt(44),l=BigInt(88),a=e*e*e%t,c=a*a*e%t,u=Jf(c,mf)*c%t,d=Jf(u,mf)*c%t,f=Jf(d,gf)*a%t,p=Jf(f,i)*f%t,h=Jf(p,r)*p%t,g=Jf(h,o)*h%t,m=Jf(g,l)*g%t,y=Jf(m,o)*h%t,v=Jf(y,mf)*c%t,w=Jf(v,s)*p%t,A=Jf(w,n)*a%t,b=Jf(A,gf);if(b*b%t!==e)throw new Error("Cannot find square root");return b}(kf(n));const r=(i&hf)===hf;if(t)r&&(i=Gf(-i));else{!(1&~e[0])!==r&&(i=Gf(-i))}const s=new Bf(n,i);return s.assertValidity(),s}static fromUncompressedHex(e){const t=zf(e.subarray(1,33)),n=zf(e.subarray(33,65)),i=new Bf(t,n);return i.assertValidity(),i}static fromHex(e){const t=jf(e),n=t.length,i=t[0];if(n===bf)return this.fromCompressedHex(t);if(33===n&&(2===i||3===i))return this.fromCompressedHex(t);if(65===n&&4===i)return this.fromUncompressedHex(t);throw new Error(`Point.fromHex: received invalid point. Expected 32-33 compressed bytes or 65 uncompressed bytes, not ${n}`)}static fromPrivateKey(e){return Bf.BASE.multiply(function(e){let t;if("bigint"==typeof e)t=e;else if("number"==typeof e&&Number.isSafeInteger(e)&&e>0)t=BigInt(e);else if("string"==typeof e){if(64!==e.length)throw new Error("Expected 32 bytes of private key");t=Mf(e)}else{if(!$f(e))throw new TypeError("Expected valid private key");if(32!==e.length)throw new Error("Expected 32 bytes of private key");t=zf(e)}if(!Vf(t))throw new Error("Expected private key: 0 < key < n");return t}(e))}static fromSignature(e,t,n){const{r:i,s:r}=function(e){if(e instanceof Df)return e.assertValidity(),e;try{return Df.fromDER(e)}catch(t){return Df.fromCompact(e)}}(t);if(![0,1,2,3].includes(n))throw new Error("Cannot recover: invalid recovery bit");const s=function(e,t=!1){const n=function(e){const t=8*e.length-256,n=zf(e);return t>0?n>>BigInt(t):n}(e);if(t)return n;const{n:i}=vf;return n>=i?n-i:n}(jf(e)),{n:o}=vf,l=2===n||3===n?i+o:i,a=Kf(l,o),c=Gf(-s*a,o),u=Gf(r*a,o),d=1&n?"03":"02",f=Bf.fromHex(d+Lf(l)),p=Bf.BASE.multiplyAndAddUnsafe(f,c,u);if(!p)throw new Error("Cannot recover signature: point at infinify");return p.assertValidity(),p}toRawBytes(e=!1){return _f(this.toHex(e))}toHex(e=!1){const t=Lf(this.x);if(e){return`${this.hasEvenY()?"02":"03"}${t}`}return`04${t}${Lf(this.y)}`}toHexX(){return this.toHex(!0).slice(2)}toRawX(){return this.toRawBytes(!0).slice(1)}assertValidity(){const e="Point is not on elliptic curve",{x:t,y:n}=this;if(!qf(t)||!qf(n))throw new Error(e);const i=Gf(n*n);if(Gf(i-kf(t))!==pf)throw new Error(e)}equals(e){return this.x===e.x&&this.y===e.y}negate(){return new Bf(this.x,Gf(-this.y))}double(){return Ef.fromAffine(this).double().toAffine()}add(e){return Ef.fromAffine(this).add(Ef.fromAffine(e)).toAffine()}subtract(e){return this.add(e.negate())}multiply(e){return Ef.fromAffine(this).multiply(e,this).toAffine()}multiplyAndAddUnsafe(e,t,n){const i=Ef.fromAffine(this),r=t===pf||t===hf||this!==Bf.BASE?i.multiplyUnsafe(t):i.multiply(t),s=Ef.fromAffine(e).multiplyUnsafe(n),o=r.add(s);return o.equals(Ef.ZERO)?void 0:o.toAffine()}}function Qf(e){return Number.parseInt(e[0],16)>=8?"00"+e:e}function Ff(e){if(e.length<2||2!==e[0])throw new Error(`Invalid signature integer tag: ${Rf(e)}`);const t=e[1],n=e.subarray(2,t+2);if(!t||n.length!==t)throw new Error("Invalid signature integer: wrong length");if(0===n[0]&&n[1]<=127)throw new Error("Invalid signature integer: trailing length");return{data:zf(n),left:e.subarray(t+2)}}Bf.BASE=new Bf(vf.Gx,vf.Gy),Bf.ZERO=new Bf(pf,pf);class Df{constructor(e,t){this.r=e,this.s=t,this.assertValidity()}static fromCompact(e){const t=$f(e),n="Signature.fromCompact";if("string"!=typeof e&&!t)throw new TypeError(`${n}: Expected string or Uint8Array`);const i=t?Rf(e):e;if(128!==i.length)throw new Error(`${n}: Expected 64-byte hex`);return new Df(Mf(i.slice(0,64)),Mf(i.slice(64,128)))}static fromDER(e){const t=$f(e);if("string"!=typeof e&&!t)throw new TypeError("Signature.fromDER: Expected string or Uint8Array");const{r:n,s:i}=function(e){if(e.length<2||48!=e[0])throw new Error(`Invalid signature tag: ${Rf(e)}`);if(e[1]!==e.length-2)throw new Error("Invalid signature: incorrect length");const{data:t,left:n}=Ff(e.subarray(2)),{data:i,left:r}=Ff(n);if(r.length)throw new Error(`Invalid signature: left bytes after parsing: ${Rf(r)}`);return{r:t,s:i}}(t?e:_f(e));return new Df(n,i)}static fromHex(e){return this.fromDER(e)}assertValidity(){const{r:e,s:t}=this;if(!Vf(e))throw new Error("Invalid Signature: r must be 0 < r < n");if(!Vf(t))throw new Error("Invalid Signature: s must be 0 < s < n")}hasHighS(){const e=vf.n>>hf;return this.s>e}normalizeS(){return this.hasHighS()?new Df(this.r,Gf(-this.s,vf.n)):this}toDERRawBytes(){return _f(this.toDERHex())}toDERHex(){const e=Qf(Of(this.s)),t=Qf(Of(this.r)),n=e.length/2,i=t.length/2,r=Of(n),s=Of(i);return`30${Of(i+n+4)}02${s}${t}02${r}${e}`}toRawBytes(){return this.toDERRawBytes()}toHex(){return this.toDERHex()}toCompactRawBytes(){return _f(this.toCompactHex())}toCompactHex(){return Lf(this.r)+Lf(this.s)}}function $f(e){return e instanceof Uint8Array||ArrayBuffer.isView(e)&&"Uint8Array"===e.constructor.name}const Pf=Array.from({length:256},(e,t)=>t.toString(16).padStart(2,"0"));function Rf(e){!function(e){if(!$f(e))throw new Error("Uint8Array expected")}(e);let t="";for(let n=0;n=Tf._0&&e<=Tf._9?e-Tf._0:e>=Tf.A&&e<=Tf.F?e-(Tf.A-10):e>=Tf.a&&e<=Tf.f?e-(Tf.a-10):void 0}function _f(e){if("string"!=typeof e)throw new Error("hex string expected, got "+typeof e);const t=e.length,n=t/2;if(t%2)throw new Error("hex string expected, got unpadded hex of length "+t);const i=new Uint8Array(n);for(let t=0,r=0;t0)return BigInt(e);if("bigint"==typeof e&&Vf(e))return e;throw new TypeError("Expected valid private scalar: 0 < scalar < curve.n")}function Gf(e,t=vf.P){const n=e%t;return n>=pf?n:t+n}function Jf(e,t){const{P:n}=vf;let i=e;for(;t-- >pf;)i*=i,i%=n;return i}function Kf(e,t=vf.P){if(e===pf||t<=pf)throw new Error(`invert: expected positive integers, got n=${e} mod=${t}`);let n=Gf(e,t),i=t,r=pf,s=hf;for(;n!==pf;){const e=i%n,t=r-s*(i/n);i=n,n=e,r=s,s=t}if(i!==hf)throw new Error("invert: does not exist");return Gf(r,t)}function Vf(e){return pf"string"==typeof e):t.every(e=>Number.isSafeInteger(e))))}function Zf(e,t){if("string"!=typeof t)throw new Error(`${e}: string expected`);return!0}function Xf(e){if(!Number.isSafeInteger(e))throw new Error(`invalid integer: ${e}`)}function ep(e){if(!Array.isArray(e))throw new Error("array expected")}function tp(e,t){if(!Wf(!0,t))throw new Error(`${e}: array of strings expected`)}Bf.BASE._setWindowSize(8);const np=(e,t)=>0===t?e:np(t,e%t),ip=(e,t)=>e+(t-np(e,t)),rp=(()=>{let e=[];for(let t=0;t<40;t++)e.push(2**t);return e})();function sp(e,t,n,i){if(ep(e),t<=0||t>32)throw new Error(`convertRadix2: wrong from=${t}`);if(n<=0||n>32)throw new Error(`convertRadix2: wrong to=${n}`);if(ip(t,n)>32)throw new Error(`convertRadix2: carry overflow from=${t} to=${n} carryBits=${ip(t,n)}`);let r=0,s=0;const o=rp[t],l=rp[n]-1,a=[];for(const i of e){if(Xf(i),i>=o)throw new Error(`convertRadix2: invalid data word=${i} from=${t}`);if(r=r<32)throw new Error(`convertRadix2: carry overflow pos=${s} from=${t}`);for(s+=t;s>=n;s-=n)a.push((r>>s-n&l)>>>0);const e=rp[s];if(void 0===e)throw new Error("invalid carry");r&=e-1}if(r=r<=t)throw new Error("Excess padding");if(!i&&r>0)throw new Error(`Non-zero padding: ${r}`);return i&&s>0&&a.push(r>>>0),a}const op=(()=>"function"==typeof Uint8Array.from([]).toBase64&&"function"==typeof Uint8Array.fromBase64)()?{encode:e=>(function(e,...t){if(!Yf(e))throw new Error("Uint8Array expected");if(t.length>0&&!t.includes(e.length))throw new Error("Uint8Array expected of length "+t+", got length="+e.length)}(e),e.toBase64()),decode:e=>((e,t)=>{Zf("base64",e);const n=t?/^[A-Za-z0-9=_-]+$/:/^[A-Za-z0-9=+/]+$/,i=t?"base64url":"base64";if(e.length>0&&!n.test(e))throw new Error("invalid base64");return Uint8Array.fromBase64(e,{alphabet:i,lastChunkHandling:"strict"})})(e,!1)}:function(...e){const t=e=>e,n=(e,t)=>n=>e(t(n));return{encode:e.map(e=>e.encode).reduceRight(n,t),decode:e.map(e=>e.decode).reduce(n,t)}}(function(e,t=!1){if(Xf(e),e<=0||e>32)throw new Error("radix2: bits should be in (0..32]");if(ip(8,e)>32||ip(e,8)>32)throw new Error("radix2: carry overflow");return{encode:n=>{if(!Yf(n))throw new Error("radix2.encode input should be Uint8Array");return sp(Array.from(n),8,e,!t)},decode:n=>(function(e,t){if(!Wf(!1,t))throw new Error(`${e}: array of numbers expected`)}("radix2.decode",n),Uint8Array.from(sp(n,e,8,t)))}}(6),function(e){const t="string"==typeof e?e.split(""):e,n=t.length;tp("alphabet",t);const i=new Map(t.map((e,t)=>[e,t]));return{encode:i=>(ep(i),i.map(i=>{if(!Number.isSafeInteger(i)||i<0||i>=n)throw new Error(`alphabet.encode: digit index outside alphabet "${i}". Allowed: ${e}`);return t[i]})),decode:t=>(ep(t),t.map(t=>{Zf("alphabet.decode",t);const n=i.get(t);if(void 0===n)throw new Error(`Unknown letter: "${t}". Allowed: ${e}`);return n}))}}("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"),function(e,t="="){return Xf(e),Zf("padding",t),{encode(n){for(tp("padding.encode",n);n.length*e%8;)n.push(t);return n},decode(n){tp("padding.decode",n);let i=n.length;if(i*e%8)throw new Error("padding: invalid, string should have whole number of bytes");for(;i>0&&n[i-1]===t;i--){if((i-1)*e%8==0)throw new Error("padding: invalid, string has too much padding")}return n.slice(0,i)}}}(6),function(e=""){return Zf("join",e),{encode:t=>(tp("join.decode",t),t.join(e)),decode:t=>(Zf("join.decode",t),t.split(e))}}(""));function lp(e){return Bf.fromHex(e).toHex().substring(2)}const ap=new TextDecoder("utf-8"),cp=new TextEncoder;class up{log=gd.extend("SerialPortSigner");writer=null;pubkey;get isConnected(){return!!this.writer}verifyEvent=fr;nip04;constructor(){this.nip04={encrypt:this.nip04Encrypt.bind(this),decrypt:this.nip04Decrypt.bind(this)}}lastCommand=null;async callMethodOnDevice(e,t,n={}){if(!up.SUPPORTED)throw new Error("Serial devices are not supported");if(this.writer||await this.connectToDevice(n),this.lastCommand)throw new Error("Previous command to device still pending!");const i=cf();return this.lastCommand=i,this.sendCommand(e,t),setTimeout(()=>{i.reject(new Error("Device timeout")),this.lastCommand===i&&(this.lastCommand=null)},6e3),this.lastCommand}async connectToDevice({onConnect:e,onDisconnect:t,onError:n,onDone:i}){let r,s=await window.navigator.serial.requestPort();await s.open({baudRate:9600}),await(e=>new Promise(t=>setTimeout(t,e)))(1e3),(async()=>{for(;s&&s.readable;){const e=new window.TextDecoderStream;s.readable.pipeTo(e.writable),r=e.readable.getReader();const t=this.readFromSerialPort(r);try{for(;;){const{value:e,done:n}=await t("\n");if(e){const{method:t,data:n}=this.parseResponse(e);if("/ping"===t&&this.log("Pong"),-1===up.PUBLIC_METHODS.indexOf(t))continue;this.log("Received: ",t,n),this.lastCommand&&(this.lastCommand.resolve(n),this.lastCommand=null)}if(n)return this.lastCommand=null,this.writer=null,void(i&&i())}}catch(e){if(e instanceof Error)throw this.writer=null,n&&n(e),this.lastCommand&&(this.lastCommand.reject(e),this.lastCommand=null),e}}})();const o=new window.TextEncoderStream;o.readable.pipeTo(s.writable),this.writer=o.writable.getWriter(),await this.sendCommand(up.METHOD_PING),await this.sendCommand(up.METHOD_PING,[window.location.host]),e&&e(),s.addEventListener("disconnect",()=>{this.log("Disconnected"),this.lastCommand=null,this.writer=null,t&&t()})}async sendCommand(e,t=[]){if(!this.writer)return;this.log("Send command",e,t);const n=[e].concat(t).join(" ");await this.writer.write(n+"\n")}readFromSerialPort(e){let t,n=[];return async(i="\n")=>{if(n.length)return{value:n.shift().trim(),done:!1};const r=[];for(t&&(r.push(t),t=void 0);;){const{value:s,done:o}=await e.read();if(s){const e=s.split(i);if(e.length>1)return r.push(e.shift()),t=e.pop(),n=e,{value:r.join("").trim(),done:!1};r.push(s)}if(o)return{value:r.join("").trim(),done:!0}}}}parseResponse(e){const t=e.split(" ")[0];return{method:t,data:e.substring(t.length).trim()}}async nip04Encrypt(e,t){const n=Lu(await this.callMethodOnDevice(up.METHOD_SHARED_SECRET,[lp(e)]));let i=Uint8Array.from(function(e=32){if(Su&&"function"==typeof Su.getRandomValues)return Su.getRandomValues(new Uint8Array(e));if(Su&&"function"==typeof Su.randomBytes)return Uint8Array.from(Su.randomBytes(e));throw new Error("crypto.getRandomValues must be defined")}(16)),r=cp.encode(t),s=await crypto.subtle.importKey("raw",n,{name:"AES-CBC"},!1,["encrypt"]),o=await crypto.subtle.encrypt({name:"AES-CBC",iv:i},s,r);return`${op.encode(new Uint8Array(o))}?iv=${op.encode(new Uint8Array(i.buffer))}`}async nip04Decrypt(e,t){let[n,i]=t.split("?iv=");const r=Lu(await this.callMethodOnDevice(up.METHOD_SHARED_SECRET,[lp(e)]));let s=await crypto.subtle.importKey("raw",r,{name:"AES-CBC"},!1,["decrypt"]),o=op.decode(n),l=op.decode(i),a=await crypto.subtle.decrypt({name:"AES-CBC",iv:l},s,o);return ap.decode(a)}async getPublicKey(){const e=await this.callMethodOnDevice(up.METHOD_PUBLIC_KEY,[]);return this.pubkey=e,e}async restore(e){await this.callMethodOnDevice(up.METHOD_RESTORE,[Du(e)])}async signEvent(e){const t=e.pubkey||this.pubkey;if(!t)throw new Error("Unknown signer pubkey");const n={...e,id:lr({...e,pubkey:t})},i=await this.callMethodOnDevice(up.METHOD_SIGN_MESSAGE,[n.id]),r={...n,sig:i,pubkey:t};if(!this.verifyEvent(r))throw new Error("Invalid signature");return r}ping(){this.sendCommand(up.METHOD_PING,[window.location.host])}static SUPPORTED="navigator"in globalThis&&!!navigator.serial;static METHOD_PING="/ping";static METHOD_LOG="/log";static METHOD_SIGN_MESSAGE="/sign-message";static METHOD_SHARED_SECRET="/shared-secret";static METHOD_PUBLIC_KEY="/public-key";static METHOD_RESTORE="/restore";static PUBLIC_METHODS=[up.METHOD_PUBLIC_KEY,up.METHOD_SIGN_MESSAGE,up.METHOD_SHARED_SECRET,up.METHOD_RESTORE]}const dp=[`${window.location.protocol.startsWith("https")?"wss:":"ws:"}//${window.location.host}/`],fp=["wss://relay.damus.io","wss://nos.lol","wss://relay.nostr.band","wss://purplepag.es"],pp=[{value:0,label:"User Metadata (0)"},{value:3,label:"Follows (3)"},{value:1e4,label:"Mute list (10000)"},{value:10001,label:"Pin list (10001)"},{value:10002,label:"Relay List Metadata (10002)"},{value:10003,label:"Bookmark list (10003)"},{value:10004,label:"Communities list (10004)"},{value:10005,label:"Public chats list (10005)"},{value:10006,label:"Blocked relays list (10006)"},{value:10007,label:"Search relays list (10007)"},{value:10009,label:"User groups (10009)"},{value:10012,label:"Favorite relays list (10012)"},{value:10013,label:"Private event relay list (10013)"},{value:10015,label:"Interests list (10015)"},{value:10019,label:"Nutzap Mint Recommendation (10019)"},{value:10020,label:"Media follows (10020)"},{value:10030,label:"User emoji list (10030)"},{value:10050,label:"Relay list to receive DMs (10050)"},{value:10051,label:"KeyPackage Relays List (10051)"},{value:10063,label:"User server list (10063)"},{value:10096,label:"File storage server list (10096)"},{value:10166,label:"Relay Monitor Announcement (10166)"},{value:10312,label:"Room Presence (10312)"},{value:10377,label:"Proxy Announcement (10377)"},{value:11111,label:"Transport Method Announcement (11111)"},{value:13194,label:"Wallet Info (13194)"},{value:17375,label:"Cashu Wallet Event (17375)"},{value:3e4,label:"Follow sets (30000)"},{value:30001,label:"Generic lists (30001)"},{value:30002,label:"Relay sets (30002)"},{value:30003,label:"Bookmark sets (30003)"},{value:30004,label:"Curation sets (30004)"},{value:30005,label:"Video sets (30005)"},{value:30007,label:"Kind mute sets (30007)"},{value:30008,label:"Profile Badges (30008)"},{value:30009,label:"Badge Definition (30009)"},{value:30015,label:"Interest sets (30015)"},{value:30017,label:"Create or update a stall (30017)"},{value:30018,label:"Create or update a product (30018)"},{value:30019,label:"Marketplace UI/UX (30019)"},{value:30020,label:"Product sold as an auction (30020)"},{value:30023,label:"Long-form Content (30023)"},{value:30024,label:"Draft Long-form Content (30024)"},{value:30030,label:"Emoji sets (30030)"},{value:30040,label:"Curated Publication Index (30040)"},{value:30041,label:"Curated Publication Content (30041)"},{value:30063,label:"Release artifact sets (30063)"},{value:30078,label:"Application-specific Data (30078)"},{value:30166,label:"Relay Discovery (30166)"},{value:30267,label:"App curation sets (30267)"},{value:30311,label:"Live Event (30311)"},{value:30312,label:"Interactive Room (30312)"},{value:30313,label:"Conference Event (30313)"},{value:30315,label:"User Statuses (30315)"},{value:30388,label:"Slide Set (30388)"},{value:30402,label:"Classified Listing (30402)"},{value:30403,label:"Draft Classified Listing (30403)"},{value:30617,label:"Repository announcements (30617)"},{value:30618,label:"Repository state announcements (30618)"},{value:30818,label:"Wiki article (30818)"},{value:30819,label:"Redirects (30819)"},{value:31234,label:"Draft Event (31234)"},{value:31388,label:"Link Set (31388)"},{value:31890,label:"Feed (31890)"},{value:31922,label:"Date-Based Calendar Event (31922)"},{value:31923,label:"Time-Based Calendar Event (31923)"},{value:31924,label:"Calendar (31924)"},{value:31925,label:"Calendar Event RSVP (31925)"},{value:31989,label:"Handler recommendation (31989)"},{value:31990,label:"Handler information (31990)"},{value:32267,label:"Software Application (32267)"},{value:34550,label:"Community Definition (34550)"},{value:37516,label:"Geocache listing (37516)"},{value:38172,label:"Cashu Mint Announcement (38172)"},{value:38173,label:"Fedimint Announcement (38173)"},{value:38383,label:"Peer-to-peer Order events (38383)"},{value:39089,label:"Starter packs (39089)"},{value:39092,label:"Media starter packs (39092)"},{value:39701,label:"Web bookmarks (39701)"}],hp={0:"User Metadata",1:"Short Text Note",2:"Recommend Relay",3:"Follows",4:"Encrypted Direct Messages",5:"Event Deletion Request",6:"Repost",7:"Reaction",8:"Badge Award",9:"Chat Message",10:"Group Chat Threaded Reply",11:"Thread",12:"Group Thread Reply",13:"Seal",14:"Direct Message",15:"File Message",16:"Generic Repost",17:"Reaction to a website",20:"Picture",40:"Channel Creation",41:"Channel Metadata",42:"Channel Message",43:"Channel Hide Message",44:"Channel Mute User",1021:"Bid",1022:"Bid Confirmation",1040:"OpenTimestamps",1063:"File Metadata",1311:"Live Chat Message",1971:"Problem Tracker",1984:"Reporting",1985:"Label",4550:"Community Post Approval",5e3:"Job Request",5999:"Job Request",6e3:"Job Result",6999:"Job Result",7e3:"Job Feedback",9041:"Zap Goal",9734:"Zap Request",9735:"Zap",9882:"Highlights",1e4:"Mute list",10001:"Pin list",10002:"Relay List Metadata",10003:"Bookmarks list",10004:"Communities list",10005:"Public Chats list",10006:"Blocked Relays list",10007:"Search Relays list",10015:"Interests",10030:"User Emoji list",10050:"DM relays",10096:"File Storage Server List",13194:"Wallet Service Info",21e3:"Lightning pub RPC",22242:"Client Authentication",23194:"Wallet Request",23195:"Wallet Response",23196:"Wallet Notification",23197:"Wallet Notification",24133:"Nostr Connect",27235:"HTTP Auth",3e4:"Follow sets",30001:"Generic lists",30002:"Relay sets",30003:"Bookmark sets",30004:"Curation sets",30008:"Profile Badges",30009:"Badge Definition",30015:"Interest sets",30017:"Stall Definition",30018:"Product Definition",30019:"Marketplace UI/UX",30020:"Product sold as an auction",30023:"Long-form Content",30024:"Draft Long-form Content",30030:"Emoji sets",30078:"Application-specific Data",30311:"Live Event",30315:"User Statuses",30402:"Classified Listing",30403:"Draft Classified Listing",31922:"Date-Based Calendar Event",31923:"Time-Based Calendar Event",31924:"Calendar",31925:"Calendar Event RSVP",31989:"Handler recommendation",31990:"Handler information",34235:"Video Event Horizontal",34236:"Video Event Vertical",34550:"Community Definition"},gp=3e5;class mp{constructor(){this.pool=new Tn,this.eventStore=new af,this.isConnected=!1,this.signer=null,this.relays=[...dp]}async connect(){console.log("Starting connection to",this.relays.length,"relays...");try{this.isConnected=!0,console.log("✓ Successfully initialized relay pool"),await new Promise(e=>setTimeout(e,1e3))}catch(e){throw console.error("✗ Connection failed:",e),e}}async connectToRelay(e){console.log(`Adding relay: ${e}`);try{return this.relays.includes(e)||this.relays.push(e),console.log(`✓ Successfully added relay ${e}`),!0}catch(t){return console.error(`✗ Failed to add relay ${e}:`,t),!1}}subscribe(e,t){console.log("Creating subscription with filters:",e);const n=this.pool.subscribeMany(this.relays,e,{onevent(e){console.log("Event received:",e),t(e)},oneose(){console.log("EOSE received"),window.dispatchEvent(new CustomEvent("nostr-eose",{detail:{subscriptionId:n.id}}))}});return n}unsubscribe(e){console.log("Closing subscription"),e&&e.close&&e.close()}disconnect(){console.log("Disconnecting relay pool"),this.pool&&this.pool.close(this.relays),this.isConnected=!1}async publish(e,t=null){this.isConnected||(console.warn("Not connected to any relays, attempting to connect first"),await this.connect());try{const n=t||this.relays,i=this.pool.publish(n,e);return await Promise.allSettled(i),console.log("✓ Event published successfully"),await Ip([e]),console.log("Event stored in IndexedDB"),{success:!0,okCount:1,errorCount:0}}catch(e){throw console.error("✗ Failed to publish event:",e),e}}getPool(){return this.pool}getEventStore(){return this.eventStore}getSigner(){return this.signer}setSigner(e){this.signer=e}}const yp=new mp,vp="nostrCache",wp=2,Ap="events";function bp(){return new Promise((e,t)=>{try{const n=indexedDB.open(vp,wp);n.onupgradeneeded=e=>{const t=n.result;e.oldVersion;let i;i=t.objectStoreNames.contains(Ap)?n.transaction.objectStore(Ap):t.createObjectStore(Ap,{keyPath:"id"}),i.indexNames.contains("byKindAuthor")||i.createIndex("byKindAuthor",["kind","pubkey"],{unique:!1}),i.indexNames.contains("byKindAuthorCreated")||i.createIndex("byKindAuthorCreated",["kind","pubkey","created_at"],{unique:!1}),i.indexNames.contains("byKind")||i.createIndex("byKind","kind",{unique:!1}),i.indexNames.contains("byAuthor")||i.createIndex("byAuthor","pubkey",{unique:!1}),i.indexNames.contains("byCreatedAt")||i.createIndex("byCreatedAt","created_at",{unique:!1})},n.onsuccess=()=>e(n.result),n.onerror=()=>t(n.error)}catch(e){console.error("Failed to open IndexedDB",e),t(e)}})}async function kp(e){try{const t=await bp();await new Promise((n,i)=>{const r=t.transaction(Ap,"readwrite");r.oncomplete=()=>n(),r.onerror=()=>i(r.error),r.objectStore(Ap).put(e)})}catch(e){console.warn("IDB putEvent failed",e)}}async function Ip(e){if(e&&0!==e.length)try{const t=await bp();await new Promise((n,i)=>{const r=t.transaction(Ap,"readwrite");r.oncomplete=()=>n(),r.onerror=()=>i(r.error);const s=r.objectStore(Ap);for(const t of e)s.put(t)}),console.log(`Stored ${e.length} events in IndexedDB`)}catch(e){console.warn("IDB putEvents failed",e)}}async function Cp(e){try{const t=await bp(),n=[];console.log("QueryEventsFromDB: Starting query with filters:",e);for(const i of e){console.log("QueryEventsFromDB: Processing filter:",i);const e=await new Promise((e,n)=>{const r=t.transaction(Ap,"readonly").objectStore(Ap),s=[];let o;if(i.kinds&&i.kinds.length>0&&i.authors&&i.authors.length>0){const e=i.kinds[0],t=i.authors[0];console.log(`QueryEventsFromDB: Using byKindAuthorCreated index for kind=${e}, author=${t.substring(0,8)}...`);const n=r.index("byKindAuthorCreated"),s=IDBKeyRange.bound([e,t,-1/0],[e,t,1/0]);o=n.openCursor(s,"prev")}else if(i.kinds&&i.kinds.length>0){console.log(`QueryEventsFromDB: Using byKind index for kind=${i.kinds[0]}`);const e=r.index("byKind");o=e.openCursor(IDBKeyRange.only(i.kinds[0]))}else if(i.authors&&i.authors.length>0){console.log(`QueryEventsFromDB: Using byAuthor index for author=${i.authors[0].substring(0,8)}...`);const e=r.index("byAuthor");o=e.openCursor(IDBKeyRange.only(i.authors[0]))}else console.log("QueryEventsFromDB: Scanning all events (no specific index)"),o=r.openCursor();o.onsuccess=t=>{const n=t.target.result;if(n){const t=n.value;let r=!0;if(i.kinds&&i.kinds.length>0&&!i.kinds.includes(t.kind)&&(r=!1),i.authors&&i.authors.length>0&&!i.authors.includes(t.pubkey)&&(r=!1),i.since&&t.created_ati.until&&(r=!1),i.ids&&i.ids.length>0&&!i.ids.includes(t.id)&&(r=!1),r&&s.push(t),i.limit&&s.length>=i.limit)return console.log(`QueryEventsFromDB: Reached limit of ${i.limit}, found ${s.length} matching events`),void e(s);n.continue()}else console.log(`QueryEventsFromDB: Cursor exhausted, found ${s.length} matching events`),e(s)},o.onerror=()=>{console.error("QueryEventsFromDB: Cursor error:",o.error),n(o.error)}});console.log(`QueryEventsFromDB: Found ${e.length} events for this filter`),n.push(...e)}return n.sort((e,t)=>t.created_at-e.created_at),console.log(`QueryEventsFromDB: Returning ${n.length} total events`),n}catch(e){return console.error("QueryEventsFromDB failed:",e),[]}}function Ep(e){try{const t=JSON.parse(e.content||"{}");return{name:t.name||t.display_name||"",picture:t.picture||"",banner:t.banner||"",about:t.about||"",nip05:t.nip05||"",lud16:t.lud16||t.lud06||""}}catch(e){return{name:"",picture:"",banner:"",about:"",nip05:"",lud16:""}}}async function xp(e){console.log(`Starting profile fetch for pubkey: ${e}`);try{const t=await async function(e){try{const t=await bp();return await new Promise((n,i)=>{const r=t.transaction(Ap,"readonly").objectStore(Ap).index("byKindAuthorCreated"),s=IDBKeyRange.bound([0,e,-1/0],[0,e,1/0]),o=r.openCursor(s,"prev");o.onsuccess=()=>{const e=o.result;n(e?e.value:null)},o.onerror=()=>i(o.error)})}catch(e){return console.warn("IDB getLatestProfileEvent failed",e),null}}(e);if(t){console.log("Using cached profile event");return Ep(t)}}catch(e){console.warn("Failed to load cached profile",e)}const t=[{kinds:[0],authors:[e],limit:1}];try{const n=await Bp(t,{timeout:1e4});if(n.length>0){const t=n[0];return console.log("Profile fetched from local relay:",t),Sp(t,e)}}catch(e){console.warn("Failed to fetch profile from local relay:",e)}console.log("Profile not found on local relay, trying fallback relays:",fp);try{const n=await async function(e,t){return new Promise(e=>{const n=[],i=setTimeout(()=>{r.close(),n.length>0?(n.sort((e,t)=>t.created_at-e.created_at),e(n[0])):e(null)},5e3),r=yp.pool.subscribeMany(fp,t,{onevent(e){console.log("Profile event received from fallback relay:",e.id?.substring(0,8)),n.push(e)},oneose(){clearTimeout(i),r.close(),n.length>0?(n.sort((e,t)=>t.created_at-e.created_at),e(n[0])):e(null)}})})}(0,t);if(n)return Sp(n,e)}catch(e){console.warn("Failed to fetch profile from fallback relays:",e)}console.log("No profile found for pubkey, creating default:",e);try{return await async function(e){const t=e.slice(0,6),n=`testuser${t}`,i=`${window.location.origin}/orly.png`,r={name:n,display_name:n,picture:i,about:"New ORLY user"},s={name:n,displayName:n,picture:i,about:"New ORLY user",pubkey:e};if(yp.signer)try{const e={kind:0,content:JSON.stringify(r),tags:[],created_at:Math.floor(Date.now()/1e3)},t=await yp.signer.signEvent(e);await yp.publish(t),console.log("Default profile published:",t.id)}catch(e){console.warn("Failed to publish default profile:",e)}return s}(e)}catch(e){return console.error("Failed to create default profile:",e),null}}async function Sp(e,t){await kp(e);try{console.log("Publishing profile event to local relay:",e.id),await yp.publish(e),console.log("Profile event successfully saved to local relay")}catch(e){console.warn("Failed to publish profile to local relay:",e)}const n=Ep(e);try{"undefined"!=typeof window&&window.dispatchEvent&&window.dispatchEvent(new CustomEvent("profile-updated",{detail:{pubkey:t,profile:n,event:e}}))}catch(e){console.warn("Failed to dispatch profile-updated event",e)}return n}async function Bp(e,t={}){console.log("Starting event fetch with filters:",JSON.stringify(e,null,2)),console.log("Current relays:",yp.relays),yp.isConnected&&0!==yp.relays.length||(console.warn("Client not connected, initializing..."),await Dp());const{timeout:n=3e4,useCache:i=!0}=t;if(i)try{const t=await Cp(e);t.length>0&&console.log(`Found ${t.length} cached events in IndexedDB`)}catch(e){console.warn("Failed to query cached events",e)}return new Promise((t,i)=>{const r=[],s=setTimeout(()=>{console.log(`Timeout reached after ${n}ms, returning ${r.length} events`),sub.close(),r.length>0&&Ip(r).catch(e=>console.warn("Failed to cache events",e)),t(r)},n);try{const n=Math.random().toString(36).substring(7);console.log(`📤 REQ [${n}]:`,JSON.stringify(["REQ",n,...e],null,2));const i=yp.pool.subscribeMany(yp.relays,e,{onevent(e){console.log(`📥 EVENT received for REQ [${n}]:`,{id:e.id?.substring(0,8)+"...",kind:e.kind,pubkey:e.pubkey?.substring(0,8)+"...",created_at:e.created_at,content_preview:e.content?.substring(0,50)}),r.push(e),kp(e).catch(e=>console.warn("Failed to cache event",e))},oneose(){console.log(`✅ EOSE received for REQ [${n}], got ${r.length} events`),clearTimeout(s),i.close(),r.length>0&&Ip(r).catch(e=>console.warn("Failed to cache events",e)),t(r)}})}catch(e){clearTimeout(s),console.error("Failed to fetch events:",e),i(e)}})}async function Qp(e,t={}){const{timeout:n=1e4}=t;console.log(`Fetching event by ID: ${e}`);try{const t=[{ids:[e]}];console.log("Fetching event with filters:",t);const i=await Bp(t,{timeout:n});return console.log(`Fetched ${i.length} events`),i.length>0?i[0]:null}catch(e){throw console.error("Failed to fetch event by ID:",e),e}}async function Fp(e,t={}){const{timeout:n=1e4}=t;console.log(`Fetching delete events for target: ${e}`);try{const t=[{kinds:[5],"#e":[e]}];console.log("Fetching delete events with filters:",t);const i=await Bp(t,{timeout:n});return console.log(`Fetched ${i.length} delete events`),i}catch(e){throw console.error("Failed to fetch delete events:",e),e}}async function Dp(){await yp.connect()}async function $p(e,t={}){const{timeout:n=3e4,cacheFirst:i=!0,cacheOnly:r=!1}=t;let s=[];if(i||r)try{if(s=await Cp(e),console.log(`Found ${s.length} events in cache`),r||s.length>0)return s}catch(e){console.warn("Failed to query cache",e)}if(!r){const t=await Bp(e,{timeout:n,useCache:!1});return console.log(`Fetched ${t.length} events from relay`),t}return s}async function Pp(){try{const e=await bp(),t=e.transaction(Ap,"readonly").objectStore(Ap),n=await new Promise((e,n)=>{const i=t.getAll();i.onsuccess=()=>e(i.result),i.onerror=()=>n(i.error)}),i=n.reduce((e,t)=>(e[t.kind]=(e[t.kind]||0)+1,e),{});return console.log("===== IndexedDB Contents ====="),console.log(`Total events: ${n.length}`),console.log("Events by kind:",i),console.log("Kind 0 events:",n.filter(e=>0===e.kind)),console.log("All event IDs:",n.map(e=>({id:e.id.substring(0,8),kind:e.kind,pubkey:e.pubkey.substring(0,8)}))),console.log("=============================="),{total:n.length,byKind:i,events:n}}catch(e){return console.error("Failed to debug IndexedDB:",e),null}} +function Yf(e){return e instanceof Uint8Array||ArrayBuffer.isView(e)&&"Uint8Array"===e.constructor.name}function Wf(e,t){return!!Array.isArray(t)&&(0===t.length||(e?t.every(e=>"string"==typeof e):t.every(e=>Number.isSafeInteger(e))))}function Zf(e,t){if("string"!=typeof t)throw new Error(`${e}: string expected`);return!0}function Xf(e){if(!Number.isSafeInteger(e))throw new Error(`invalid integer: ${e}`)}function ep(e){if(!Array.isArray(e))throw new Error("array expected")}function tp(e,t){if(!Wf(!0,t))throw new Error(`${e}: array of strings expected`)}Bf.BASE._setWindowSize(8);const np=(e,t)=>0===t?e:np(t,e%t),ip=(e,t)=>e+(t-np(e,t)),rp=(()=>{let e=[];for(let t=0;t<40;t++)e.push(2**t);return e})();function sp(e,t,n,i){if(ep(e),t<=0||t>32)throw new Error(`convertRadix2: wrong from=${t}`);if(n<=0||n>32)throw new Error(`convertRadix2: wrong to=${n}`);if(ip(t,n)>32)throw new Error(`convertRadix2: carry overflow from=${t} to=${n} carryBits=${ip(t,n)}`);let r=0,s=0;const o=rp[t],l=rp[n]-1,a=[];for(const i of e){if(Xf(i),i>=o)throw new Error(`convertRadix2: invalid data word=${i} from=${t}`);if(r=r<32)throw new Error(`convertRadix2: carry overflow pos=${s} from=${t}`);for(s+=t;s>=n;s-=n)a.push((r>>s-n&l)>>>0);const e=rp[s];if(void 0===e)throw new Error("invalid carry");r&=e-1}if(r=r<=t)throw new Error("Excess padding");if(!i&&r>0)throw new Error(`Non-zero padding: ${r}`);return i&&s>0&&a.push(r>>>0),a}const op=(()=>"function"==typeof Uint8Array.from([]).toBase64&&"function"==typeof Uint8Array.fromBase64)()?{encode:e=>(function(e,...t){if(!Yf(e))throw new Error("Uint8Array expected");if(t.length>0&&!t.includes(e.length))throw new Error("Uint8Array expected of length "+t+", got length="+e.length)}(e),e.toBase64()),decode:e=>((e,t)=>{Zf("base64",e);const n=t?/^[A-Za-z0-9=_-]+$/:/^[A-Za-z0-9=+/]+$/,i=t?"base64url":"base64";if(e.length>0&&!n.test(e))throw new Error("invalid base64");return Uint8Array.fromBase64(e,{alphabet:i,lastChunkHandling:"strict"})})(e,!1)}:function(...e){const t=e=>e,n=(e,t)=>n=>e(t(n));return{encode:e.map(e=>e.encode).reduceRight(n,t),decode:e.map(e=>e.decode).reduce(n,t)}}(function(e,t=!1){if(Xf(e),e<=0||e>32)throw new Error("radix2: bits should be in (0..32]");if(ip(8,e)>32||ip(e,8)>32)throw new Error("radix2: carry overflow");return{encode:n=>{if(!Yf(n))throw new Error("radix2.encode input should be Uint8Array");return sp(Array.from(n),8,e,!t)},decode:n=>(function(e,t){if(!Wf(!1,t))throw new Error(`${e}: array of numbers expected`)}("radix2.decode",n),Uint8Array.from(sp(n,e,8,t)))}}(6),function(e){const t="string"==typeof e?e.split(""):e,n=t.length;tp("alphabet",t);const i=new Map(t.map((e,t)=>[e,t]));return{encode:i=>(ep(i),i.map(i=>{if(!Number.isSafeInteger(i)||i<0||i>=n)throw new Error(`alphabet.encode: digit index outside alphabet "${i}". Allowed: ${e}`);return t[i]})),decode:t=>(ep(t),t.map(t=>{Zf("alphabet.decode",t);const n=i.get(t);if(void 0===n)throw new Error(`Unknown letter: "${t}". Allowed: ${e}`);return n}))}}("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"),function(e,t="="){return Xf(e),Zf("padding",t),{encode(n){for(tp("padding.encode",n);n.length*e%8;)n.push(t);return n},decode(n){tp("padding.decode",n);let i=n.length;if(i*e%8)throw new Error("padding: invalid, string should have whole number of bytes");for(;i>0&&n[i-1]===t;i--){if((i-1)*e%8==0)throw new Error("padding: invalid, string has too much padding")}return n.slice(0,i)}}}(6),function(e=""){return Zf("join",e),{encode:t=>(tp("join.decode",t),t.join(e)),decode:t=>(Zf("join.decode",t),t.split(e))}}(""));function lp(e){return Bf.fromHex(e).toHex().substring(2)}const ap=new TextDecoder("utf-8"),cp=new TextEncoder;class up{log=gd.extend("SerialPortSigner");writer=null;pubkey;get isConnected(){return!!this.writer}verifyEvent=fr;nip04;constructor(){this.nip04={encrypt:this.nip04Encrypt.bind(this),decrypt:this.nip04Decrypt.bind(this)}}lastCommand=null;async callMethodOnDevice(e,t,n={}){if(!up.SUPPORTED)throw new Error("Serial devices are not supported");if(this.writer||await this.connectToDevice(n),this.lastCommand)throw new Error("Previous command to device still pending!");const i=cf();return this.lastCommand=i,this.sendCommand(e,t),setTimeout(()=>{i.reject(new Error("Device timeout")),this.lastCommand===i&&(this.lastCommand=null)},6e3),this.lastCommand}async connectToDevice({onConnect:e,onDisconnect:t,onError:n,onDone:i}){let r,s=await window.navigator.serial.requestPort();await s.open({baudRate:9600}),await(e=>new Promise(t=>setTimeout(t,e)))(1e3),(async()=>{for(;s&&s.readable;){const e=new window.TextDecoderStream;s.readable.pipeTo(e.writable),r=e.readable.getReader();const t=this.readFromSerialPort(r);try{for(;;){const{value:e,done:n}=await t("\n");if(e){const{method:t,data:n}=this.parseResponse(e);if("/ping"===t&&this.log("Pong"),-1===up.PUBLIC_METHODS.indexOf(t))continue;this.log("Received: ",t,n),this.lastCommand&&(this.lastCommand.resolve(n),this.lastCommand=null)}if(n)return this.lastCommand=null,this.writer=null,void(i&&i())}}catch(e){if(e instanceof Error)throw this.writer=null,n&&n(e),this.lastCommand&&(this.lastCommand.reject(e),this.lastCommand=null),e}}})();const o=new window.TextEncoderStream;o.readable.pipeTo(s.writable),this.writer=o.writable.getWriter(),await this.sendCommand(up.METHOD_PING),await this.sendCommand(up.METHOD_PING,[window.location.host]),e&&e(),s.addEventListener("disconnect",()=>{this.log("Disconnected"),this.lastCommand=null,this.writer=null,t&&t()})}async sendCommand(e,t=[]){if(!this.writer)return;this.log("Send command",e,t);const n=[e].concat(t).join(" ");await this.writer.write(n+"\n")}readFromSerialPort(e){let t,n=[];return async(i="\n")=>{if(n.length)return{value:n.shift().trim(),done:!1};const r=[];for(t&&(r.push(t),t=void 0);;){const{value:s,done:o}=await e.read();if(s){const e=s.split(i);if(e.length>1)return r.push(e.shift()),t=e.pop(),n=e,{value:r.join("").trim(),done:!1};r.push(s)}if(o)return{value:r.join("").trim(),done:!0}}}}parseResponse(e){const t=e.split(" ")[0];return{method:t,data:e.substring(t.length).trim()}}async nip04Encrypt(e,t){const n=Lu(await this.callMethodOnDevice(up.METHOD_SHARED_SECRET,[lp(e)]));let i=Uint8Array.from(function(e=32){if(Su&&"function"==typeof Su.getRandomValues)return Su.getRandomValues(new Uint8Array(e));if(Su&&"function"==typeof Su.randomBytes)return Uint8Array.from(Su.randomBytes(e));throw new Error("crypto.getRandomValues must be defined")}(16)),r=cp.encode(t),s=await crypto.subtle.importKey("raw",n,{name:"AES-CBC"},!1,["encrypt"]),o=await crypto.subtle.encrypt({name:"AES-CBC",iv:i},s,r);return`${op.encode(new Uint8Array(o))}?iv=${op.encode(new Uint8Array(i.buffer))}`}async nip04Decrypt(e,t){let[n,i]=t.split("?iv=");const r=Lu(await this.callMethodOnDevice(up.METHOD_SHARED_SECRET,[lp(e)]));let s=await crypto.subtle.importKey("raw",r,{name:"AES-CBC"},!1,["decrypt"]),o=op.decode(n),l=op.decode(i),a=await crypto.subtle.decrypt({name:"AES-CBC",iv:l},s,o);return ap.decode(a)}async getPublicKey(){const e=await this.callMethodOnDevice(up.METHOD_PUBLIC_KEY,[]);return this.pubkey=e,e}async restore(e){await this.callMethodOnDevice(up.METHOD_RESTORE,[Du(e)])}async signEvent(e){const t=e.pubkey||this.pubkey;if(!t)throw new Error("Unknown signer pubkey");const n={...e,id:lr({...e,pubkey:t})},i=await this.callMethodOnDevice(up.METHOD_SIGN_MESSAGE,[n.id]),r={...n,sig:i,pubkey:t};if(!this.verifyEvent(r))throw new Error("Invalid signature");return r}ping(){this.sendCommand(up.METHOD_PING,[window.location.host])}static SUPPORTED="navigator"in globalThis&&!!navigator.serial;static METHOD_PING="/ping";static METHOD_LOG="/log";static METHOD_SIGN_MESSAGE="/sign-message";static METHOD_SHARED_SECRET="/shared-secret";static METHOD_PUBLIC_KEY="/public-key";static METHOD_RESTORE="/restore";static PUBLIC_METHODS=[up.METHOD_PUBLIC_KEY,up.METHOD_SIGN_MESSAGE,up.METHOD_SHARED_SECRET,up.METHOD_RESTORE]}const dp=[`${window.location.protocol.startsWith("https")?"wss:":"ws:"}//${window.location.host}/`],fp=["wss://relay.damus.io","wss://nos.lol","wss://relay.nostr.band","wss://purplepag.es"],pp=[{value:0,label:"User Metadata (0)"},{value:3,label:"Follows (3)"},{value:1e4,label:"Mute list (10000)"},{value:10001,label:"Pin list (10001)"},{value:10002,label:"Relay List Metadata (10002)"},{value:10003,label:"Bookmark list (10003)"},{value:10004,label:"Communities list (10004)"},{value:10005,label:"Public chats list (10005)"},{value:10006,label:"Blocked relays list (10006)"},{value:10007,label:"Search relays list (10007)"},{value:10009,label:"User groups (10009)"},{value:10012,label:"Favorite relays list (10012)"},{value:10013,label:"Private event relay list (10013)"},{value:10015,label:"Interests list (10015)"},{value:10019,label:"Nutzap Mint Recommendation (10019)"},{value:10020,label:"Media follows (10020)"},{value:10030,label:"User emoji list (10030)"},{value:10050,label:"Relay list to receive DMs (10050)"},{value:10051,label:"KeyPackage Relays List (10051)"},{value:10063,label:"User server list (10063)"},{value:10096,label:"File storage server list (10096)"},{value:10166,label:"Relay Monitor Announcement (10166)"},{value:10312,label:"Room Presence (10312)"},{value:10377,label:"Proxy Announcement (10377)"},{value:11111,label:"Transport Method Announcement (11111)"},{value:13194,label:"Wallet Info (13194)"},{value:17375,label:"Cashu Wallet Event (17375)"},{value:3e4,label:"Follow sets (30000)"},{value:30001,label:"Generic lists (30001)"},{value:30002,label:"Relay sets (30002)"},{value:30003,label:"Bookmark sets (30003)"},{value:30004,label:"Curation sets (30004)"},{value:30005,label:"Video sets (30005)"},{value:30007,label:"Kind mute sets (30007)"},{value:30008,label:"Profile Badges (30008)"},{value:30009,label:"Badge Definition (30009)"},{value:30015,label:"Interest sets (30015)"},{value:30017,label:"Create or update a stall (30017)"},{value:30018,label:"Create or update a product (30018)"},{value:30019,label:"Marketplace UI/UX (30019)"},{value:30020,label:"Product sold as an auction (30020)"},{value:30023,label:"Long-form Content (30023)"},{value:30024,label:"Draft Long-form Content (30024)"},{value:30030,label:"Emoji sets (30030)"},{value:30040,label:"Curated Publication Index (30040)"},{value:30041,label:"Curated Publication Content (30041)"},{value:30063,label:"Release artifact sets (30063)"},{value:30078,label:"Application-specific Data (30078)"},{value:30166,label:"Relay Discovery (30166)"},{value:30267,label:"App curation sets (30267)"},{value:30311,label:"Live Event (30311)"},{value:30312,label:"Interactive Room (30312)"},{value:30313,label:"Conference Event (30313)"},{value:30315,label:"User Statuses (30315)"},{value:30388,label:"Slide Set (30388)"},{value:30402,label:"Classified Listing (30402)"},{value:30403,label:"Draft Classified Listing (30403)"},{value:30617,label:"Repository announcements (30617)"},{value:30618,label:"Repository state announcements (30618)"},{value:30818,label:"Wiki article (30818)"},{value:30819,label:"Redirects (30819)"},{value:31234,label:"Draft Event (31234)"},{value:31388,label:"Link Set (31388)"},{value:31890,label:"Feed (31890)"},{value:31922,label:"Date-Based Calendar Event (31922)"},{value:31923,label:"Time-Based Calendar Event (31923)"},{value:31924,label:"Calendar (31924)"},{value:31925,label:"Calendar Event RSVP (31925)"},{value:31989,label:"Handler recommendation (31989)"},{value:31990,label:"Handler information (31990)"},{value:32267,label:"Software Application (32267)"},{value:34550,label:"Community Definition (34550)"},{value:37516,label:"Geocache listing (37516)"},{value:38172,label:"Cashu Mint Announcement (38172)"},{value:38173,label:"Fedimint Announcement (38173)"},{value:38383,label:"Peer-to-peer Order events (38383)"},{value:39089,label:"Starter packs (39089)"},{value:39092,label:"Media starter packs (39092)"},{value:39701,label:"Web bookmarks (39701)"}],hp={0:"User Metadata",1:"Short Text Note",2:"Recommend Relay",3:"Follows",4:"Encrypted Direct Messages",5:"Event Deletion Request",6:"Repost",7:"Reaction",8:"Badge Award",9:"Chat Message",10:"Group Chat Threaded Reply",11:"Thread",12:"Group Thread Reply",13:"Seal",14:"Direct Message",15:"File Message",16:"Generic Repost",17:"Reaction to a website",20:"Picture",40:"Channel Creation",41:"Channel Metadata",42:"Channel Message",43:"Channel Hide Message",44:"Channel Mute User",1021:"Bid",1022:"Bid Confirmation",1040:"OpenTimestamps",1063:"File Metadata",1311:"Live Chat Message",1971:"Problem Tracker",1984:"Reporting",1985:"Label",4550:"Community Post Approval",5e3:"Job Request",5999:"Job Request",6e3:"Job Result",6999:"Job Result",7e3:"Job Feedback",9041:"Zap Goal",9734:"Zap Request",9735:"Zap",9882:"Highlights",1e4:"Mute list",10001:"Pin list",10002:"Relay List Metadata",10003:"Bookmarks list",10004:"Communities list",10005:"Public Chats list",10006:"Blocked Relays list",10007:"Search Relays list",10015:"Interests",10030:"User Emoji list",10050:"DM relays",10096:"File Storage Server List",13194:"Wallet Service Info",21e3:"Lightning pub RPC",22242:"Client Authentication",23194:"Wallet Request",23195:"Wallet Response",23196:"Wallet Notification",23197:"Wallet Notification",24133:"Nostr Connect",27235:"HTTP Auth",3e4:"Follow sets",30001:"Generic lists",30002:"Relay sets",30003:"Bookmark sets",30004:"Curation sets",30008:"Profile Badges",30009:"Badge Definition",30015:"Interest sets",30017:"Stall Definition",30018:"Product Definition",30019:"Marketplace UI/UX",30020:"Product sold as an auction",30023:"Long-form Content",30024:"Draft Long-form Content",30030:"Emoji sets",30078:"Application-specific Data",30311:"Live Event",30315:"User Statuses",30402:"Classified Listing",30403:"Draft Classified Listing",31922:"Date-Based Calendar Event",31923:"Time-Based Calendar Event",31924:"Calendar",31925:"Calendar Event RSVP",31989:"Handler recommendation",31990:"Handler information",34235:"Video Event Horizontal",34236:"Video Event Vertical",34550:"Community Definition"},gp=3e5;class mp{constructor(){this.pool=new Tn,this.eventStore=new af,this.isConnected=!1,this.signer=null,this.relays=[...dp]}async connect(){console.log("Starting connection to",this.relays.length,"relays...");try{this.isConnected=!0,console.log("✓ Successfully initialized relay pool"),await new Promise(e=>setTimeout(e,1e3))}catch(e){throw console.error("✗ Connection failed:",e),e}}async connectToRelay(e){console.log(`Adding relay: ${e}`);try{return this.relays.includes(e)||this.relays.push(e),console.log(`✓ Successfully added relay ${e}`),!0}catch(t){return console.error(`✗ Failed to add relay ${e}:`,t),!1}}subscribe(e,t){console.log("Creating subscription with filters:",e);const n=this.pool.subscribeMany(this.relays,e,{onevent(e){console.log("Event received:",e),t(e)},oneose(){console.log("EOSE received"),window.dispatchEvent(new CustomEvent("nostr-eose",{detail:{subscriptionId:n.id}}))}});return n}unsubscribe(e){console.log("Closing subscription"),e&&e.close&&e.close()}disconnect(){console.log("Disconnecting relay pool"),this.pool&&this.pool.close(this.relays),this.isConnected=!1}async publish(e,t=null){this.isConnected||(console.warn("Not connected to any relays, attempting to connect first"),await this.connect());try{const n=t||this.relays,i=this.pool.publish(n,e);return await Promise.allSettled(i),console.log("✓ Event published successfully"),await Ip([e]),console.log("Event stored in IndexedDB"),{success:!0,okCount:1,errorCount:0}}catch(e){throw console.error("✗ Failed to publish event:",e),e}}getPool(){return this.pool}getEventStore(){return this.eventStore}getSigner(){return this.signer}setSigner(e){this.signer=e}}const yp=new mp,vp="nostrCache",wp=2,Ap="events";function bp(){return new Promise((e,t)=>{try{const n=indexedDB.open(vp,wp);n.onupgradeneeded=e=>{const t=n.result;e.oldVersion;let i;i=t.objectStoreNames.contains(Ap)?n.transaction.objectStore(Ap):t.createObjectStore(Ap,{keyPath:"id"}),i.indexNames.contains("byKindAuthor")||i.createIndex("byKindAuthor",["kind","pubkey"],{unique:!1}),i.indexNames.contains("byKindAuthorCreated")||i.createIndex("byKindAuthorCreated",["kind","pubkey","created_at"],{unique:!1}),i.indexNames.contains("byKind")||i.createIndex("byKind","kind",{unique:!1}),i.indexNames.contains("byAuthor")||i.createIndex("byAuthor","pubkey",{unique:!1}),i.indexNames.contains("byCreatedAt")||i.createIndex("byCreatedAt","created_at",{unique:!1})},n.onsuccess=()=>e(n.result),n.onerror=()=>t(n.error)}catch(e){console.error("Failed to open IndexedDB",e),t(e)}})}async function kp(e){try{const t=await bp();await new Promise((n,i)=>{const r=t.transaction(Ap,"readwrite");r.oncomplete=()=>n(),r.onerror=()=>i(r.error),r.objectStore(Ap).put(e)})}catch(e){console.warn("IDB putEvent failed",e)}}async function Ip(e){if(e&&0!==e.length)try{const t=await bp();await new Promise((n,i)=>{const r=t.transaction(Ap,"readwrite");r.oncomplete=()=>n(),r.onerror=()=>i(r.error);const s=r.objectStore(Ap);for(const t of e)s.put(t)}),console.log(`Stored ${e.length} events in IndexedDB`)}catch(e){console.warn("IDB putEvents failed",e)}}async function Cp(e){try{const t=await bp(),n=[];console.log("QueryEventsFromDB: Starting query with filters:",e);for(const i of e){console.log("QueryEventsFromDB: Processing filter:",i);const e=await new Promise((e,n)=>{const r=t.transaction(Ap,"readonly").objectStore(Ap),s=[];let o;if(i.kinds&&i.kinds.length>0&&i.authors&&i.authors.length>0){const e=i.kinds[0],t=i.authors[0];console.log(`QueryEventsFromDB: Using byKindAuthorCreated index for kind=${e}, author=${t.substring(0,8)}...`);const n=r.index("byKindAuthorCreated"),s=IDBKeyRange.bound([e,t,-1/0],[e,t,1/0]);o=n.openCursor(s,"prev")}else if(i.kinds&&i.kinds.length>0){console.log(`QueryEventsFromDB: Using byKind index for kind=${i.kinds[0]}`);const e=r.index("byKind");o=e.openCursor(IDBKeyRange.only(i.kinds[0]))}else if(i.authors&&i.authors.length>0){console.log(`QueryEventsFromDB: Using byAuthor index for author=${i.authors[0].substring(0,8)}...`);const e=r.index("byAuthor");o=e.openCursor(IDBKeyRange.only(i.authors[0]))}else console.log("QueryEventsFromDB: Scanning all events (no specific index)"),o=r.openCursor();o.onsuccess=t=>{const n=t.target.result;if(n){const t=n.value;let r=!0;if(i.kinds&&i.kinds.length>0&&!i.kinds.includes(t.kind)&&(r=!1),i.authors&&i.authors.length>0&&!i.authors.includes(t.pubkey)&&(r=!1),i.since&&t.created_ati.until&&(r=!1),i.ids&&i.ids.length>0&&!i.ids.includes(t.id)&&(r=!1),r&&s.push(t),i.limit&&s.length>=i.limit)return console.log(`QueryEventsFromDB: Reached limit of ${i.limit}, found ${s.length} matching events`),void e(s);n.continue()}else console.log(`QueryEventsFromDB: Cursor exhausted, found ${s.length} matching events`),e(s)},o.onerror=()=>{console.error("QueryEventsFromDB: Cursor error:",o.error),n(o.error)}});console.log(`QueryEventsFromDB: Found ${e.length} events for this filter`),n.push(...e)}return n.sort((e,t)=>t.created_at-e.created_at),console.log(`QueryEventsFromDB: Returning ${n.length} total events`),n}catch(e){return console.error("QueryEventsFromDB failed:",e),[]}}function Ep(e){try{const t=JSON.parse(e.content||"{}");return{name:t.name||t.display_name||"",picture:t.picture||"",banner:t.banner||"",about:t.about||"",nip05:t.nip05||"",lud16:t.lud16||t.lud06||""}}catch(e){return{name:"",picture:"",banner:"",about:"",nip05:"",lud16:""}}}async function xp(e){console.log(`Starting profile fetch for pubkey: ${e}`);try{const t=await async function(e){try{const t=await bp();return await new Promise((n,i)=>{const r=t.transaction(Ap,"readonly").objectStore(Ap).index("byKindAuthorCreated"),s=IDBKeyRange.bound([0,e,-1/0],[0,e,1/0]),o=r.openCursor(s,"prev");o.onsuccess=()=>{const e=o.result;n(e?e.value:null)},o.onerror=()=>i(o.error)})}catch(e){return console.warn("IDB getLatestProfileEvent failed",e),null}}(e);if(t){console.log("Using cached profile event");return Ep(t)}}catch(e){console.warn("Failed to load cached profile",e)}const t=[{kinds:[0],authors:[e],limit:1}];try{const n=await Bp(t,{timeout:1e4});if(n.length>0){const t=n[0];return console.log("Profile fetched from local relay:",t),Sp(t,e)}}catch(e){console.warn("Failed to fetch profile from local relay:",e)}console.log("Profile not found on local relay, trying fallback relays:",fp);try{const n=await async function(e,t){return new Promise(e=>{const n=[],i=setTimeout(()=>{r.close(),n.length>0?(n.sort((e,t)=>t.created_at-e.created_at),e(n[0])):e(null)},5e3),r=yp.pool.subscribeMany(fp,t,{onevent(e){console.log("Profile event received from fallback relay:",e.id?.substring(0,8)),n.push(e)},oneose(){clearTimeout(i),r.close(),n.length>0?(n.sort((e,t)=>t.created_at-e.created_at),e(n[0])):e(null)}})})}(0,t);if(n)return Sp(n,e)}catch(e){console.warn("Failed to fetch profile from fallback relays:",e)}return console.log("No profile found for pubkey:",e),null}async function Sp(e,t){await kp(e);try{console.log("Publishing profile event to local relay:",e.id),await yp.publish(e),console.log("Profile event successfully saved to local relay")}catch(e){console.warn("Failed to publish profile to local relay:",e)}const n=Ep(e);try{"undefined"!=typeof window&&window.dispatchEvent&&window.dispatchEvent(new CustomEvent("profile-updated",{detail:{pubkey:t,profile:n,event:e}}))}catch(e){console.warn("Failed to dispatch profile-updated event",e)}return n}async function Bp(e,t={}){console.log("Starting event fetch with filters:",JSON.stringify(e,null,2)),console.log("Current relays:",yp.relays),yp.isConnected&&0!==yp.relays.length||(console.warn("Client not connected, initializing..."),await Dp());const{timeout:n=3e4,useCache:i=!0}=t;if(i)try{const t=await Cp(e);t.length>0&&console.log(`Found ${t.length} cached events in IndexedDB`)}catch(e){console.warn("Failed to query cached events",e)}return new Promise((t,i)=>{const r=[],s=setTimeout(()=>{console.log(`Timeout reached after ${n}ms, returning ${r.length} events`),sub.close(),r.length>0&&Ip(r).catch(e=>console.warn("Failed to cache events",e)),t(r)},n);try{const n=Math.random().toString(36).substring(7);console.log(`📤 REQ [${n}]:`,JSON.stringify(["REQ",n,...e],null,2));const i=yp.pool.subscribeMany(yp.relays,e,{onevent(e){console.log(`📥 EVENT received for REQ [${n}]:`,{id:e.id?.substring(0,8)+"...",kind:e.kind,pubkey:e.pubkey?.substring(0,8)+"...",created_at:e.created_at,content_preview:e.content?.substring(0,50)}),r.push(e),kp(e).catch(e=>console.warn("Failed to cache event",e))},oneose(){console.log(`✅ EOSE received for REQ [${n}], got ${r.length} events`),clearTimeout(s),i.close(),r.length>0&&Ip(r).catch(e=>console.warn("Failed to cache events",e)),t(r)}})}catch(e){clearTimeout(s),console.error("Failed to fetch events:",e),i(e)}})}async function Qp(e,t={}){const{timeout:n=1e4}=t;console.log(`Fetching event by ID: ${e}`);try{const t=[{ids:[e]}];console.log("Fetching event with filters:",t);const i=await Bp(t,{timeout:n});return console.log(`Fetched ${i.length} events`),i.length>0?i[0]:null}catch(e){throw console.error("Failed to fetch event by ID:",e),e}}async function Fp(e,t={}){const{timeout:n=1e4}=t;console.log(`Fetching delete events for target: ${e}`);try{const t=[{kinds:[5],"#e":[e]}];console.log("Fetching delete events with filters:",t);const i=await Bp(t,{timeout:n});return console.log(`Fetched ${i.length} delete events`),i}catch(e){throw console.error("Failed to fetch delete events:",e),e}}async function Dp(){await yp.connect()}async function $p(e,t={}){const{timeout:n=3e4,cacheFirst:i=!0,cacheOnly:r=!1}=t;let s=[];if(i||r)try{if(s=await Cp(e),console.log(`Found ${s.length} events in cache`),r||s.length>0)return s}catch(e){console.warn("Failed to query cache",e)}if(!r){const t=await Bp(e,{timeout:n,useCache:!1});return console.log(`Fetched ${t.length} events from relay`),t}return s}async function Pp(){try{const e=await bp(),t=e.transaction(Ap,"readonly").objectStore(Ap),n=await new Promise((e,n)=>{const i=t.getAll();i.onsuccess=()=>e(i.result),i.onerror=()=>n(i.error)}),i=n.reduce((e,t)=>(e[t.kind]=(e[t.kind]||0)+1,e),{});return console.log("===== IndexedDB Contents ====="),console.log(`Total events: ${n.length}`),console.log("Events by kind:",i),console.log("Kind 0 events:",n.filter(e=>0===e.kind)),console.log("All event IDs:",n.map(e=>({id:e.id.substring(0,8),kind:e.kind,pubkey:e.pubkey.substring(0,8)}))),console.log("=============================="),{total:n.length,byKind:i,events:n}}catch(e){return console.error("Failed to debug IndexedDB:",e),null}} /*! * hash-wasm (https://www.npmjs.com/package/hash-wasm) * (c) Dani Biro diff --git a/app/web/dist/bundle.js.map b/app/web/dist/bundle.js.map index f400fe9..7b72eea 100644 --- a/app/web/dist/bundle.js.map +++ b/app/web/dist/bundle.js.map @@ -1 +1 @@ -{"version":3,"file":"bundle.js","sources":["../node_modules/svelte/internal/index.mjs","../node_modules/nostr-tools/node_modules/@noble/curves/node_modules/@noble/hashes/esm/_assert.js","../node_modules/nostr-tools/node_modules/@noble/curves/node_modules/@noble/hashes/esm/crypto.js","../node_modules/nostr-tools/node_modules/@noble/curves/node_modules/@noble/hashes/esm/utils.js","../node_modules/nostr-tools/node_modules/@noble/curves/node_modules/@noble/hashes/esm/_sha2.js","../node_modules/nostr-tools/node_modules/@noble/curves/node_modules/@noble/hashes/esm/sha256.js","../node_modules/nostr-tools/node_modules/@noble/curves/esm/abstract/utils.js","../node_modules/nostr-tools/node_modules/@noble/curves/esm/abstract/modular.js","../node_modules/nostr-tools/node_modules/@noble/curves/esm/abstract/curve.js","../node_modules/nostr-tools/node_modules/@noble/curves/esm/abstract/weierstrass.js","../node_modules/nostr-tools/node_modules/@noble/curves/node_modules/@noble/hashes/esm/hmac.js","../node_modules/nostr-tools/node_modules/@noble/curves/esm/_shortw_utils.js","../node_modules/nostr-tools/node_modules/@noble/curves/esm/secp256k1.js","../node_modules/nostr-tools/node_modules/@noble/hashes/esm/crypto.js","../node_modules/nostr-tools/node_modules/@noble/hashes/esm/utils.js","../node_modules/nostr-tools/node_modules/@noble/hashes/esm/_assert.js","../node_modules/nostr-tools/node_modules/@noble/hashes/esm/_sha2.js","../node_modules/nostr-tools/node_modules/@noble/hashes/esm/sha256.js","../node_modules/nostr-tools/lib/esm/pool.js","../node_modules/nostr-tools/node_modules/@scure/base/lib/esm/index.js","../node_modules/@noble/ciphers/esm/_assert.js","../node_modules/@noble/ciphers/esm/utils.js","../node_modules/@noble/ciphers/esm/aes.js","../node_modules/@noble/ciphers/esm/_arx.js","../node_modules/@noble/ciphers/esm/chacha.js","../node_modules/nostr-tools/node_modules/@noble/hashes/esm/hmac.js","../node_modules/nostr-tools/node_modules/@noble/hashes/esm/hkdf.js","../node_modules/nostr-tools/lib/esm/index.js","../node_modules/nostr-tools/lib/esm/kinds.js","../node_modules/tslib/tslib.es6.mjs","../node_modules/rxjs/dist/esm5/internal/util/isFunction.js","../node_modules/rxjs/dist/esm5/internal/util/createErrorClass.js","../node_modules/rxjs/dist/esm5/internal/util/UnsubscriptionError.js","../node_modules/rxjs/dist/esm5/internal/util/arrRemove.js","../node_modules/rxjs/dist/esm5/internal/Subscription.js","../node_modules/rxjs/dist/esm5/internal/config.js","../node_modules/rxjs/dist/esm5/internal/scheduler/timeoutProvider.js","../node_modules/rxjs/dist/esm5/internal/util/reportUnhandledError.js","../node_modules/rxjs/dist/esm5/internal/util/noop.js","../node_modules/rxjs/dist/esm5/internal/util/errorContext.js","../node_modules/rxjs/dist/esm5/internal/Subscriber.js","../node_modules/rxjs/dist/esm5/internal/symbol/observable.js","../node_modules/rxjs/dist/esm5/internal/util/identity.js","../node_modules/rxjs/dist/esm5/internal/Observable.js","../node_modules/rxjs/dist/esm5/internal/util/pipe.js","../node_modules/rxjs/dist/esm5/internal/util/lift.js","../node_modules/rxjs/dist/esm5/internal/operators/OperatorSubscriber.js","../node_modules/rxjs/dist/esm5/internal/util/ObjectUnsubscribedError.js","../node_modules/rxjs/dist/esm5/internal/Subject.js","../node_modules/rxjs/dist/esm5/internal/scheduler/dateTimestampProvider.js","../node_modules/rxjs/dist/esm5/internal/ReplaySubject.js","../node_modules/rxjs/dist/esm5/internal/scheduler/Action.js","../node_modules/rxjs/dist/esm5/internal/scheduler/intervalProvider.js","../node_modules/rxjs/dist/esm5/internal/scheduler/AsyncAction.js","../node_modules/rxjs/dist/esm5/internal/Scheduler.js","../node_modules/rxjs/dist/esm5/internal/scheduler/AsyncScheduler.js","../node_modules/rxjs/dist/esm5/internal/scheduler/async.js","../node_modules/rxjs/dist/esm5/internal/observable/empty.js","../node_modules/rxjs/dist/esm5/internal/util/isScheduler.js","../node_modules/rxjs/dist/esm5/internal/util/args.js","../node_modules/rxjs/dist/esm5/internal/util/isArrayLike.js","../node_modules/rxjs/dist/esm5/internal/util/isPromise.js","../node_modules/rxjs/dist/esm5/internal/util/isInteropObservable.js","../node_modules/rxjs/dist/esm5/internal/util/isAsyncIterable.js","../node_modules/rxjs/dist/esm5/internal/util/throwUnobservableError.js","../node_modules/rxjs/dist/esm5/internal/symbol/iterator.js","../node_modules/rxjs/dist/esm5/internal/util/isIterable.js","../node_modules/rxjs/dist/esm5/internal/util/isReadableStreamLike.js","../node_modules/rxjs/dist/esm5/internal/observable/innerFrom.js","../node_modules/rxjs/dist/esm5/internal/util/executeSchedule.js","../node_modules/rxjs/dist/esm5/internal/operators/observeOn.js","../node_modules/rxjs/dist/esm5/internal/operators/subscribeOn.js","../node_modules/rxjs/dist/esm5/internal/scheduled/scheduleAsyncIterable.js","../node_modules/rxjs/dist/esm5/internal/scheduled/scheduled.js","../node_modules/rxjs/dist/esm5/internal/scheduled/scheduleObservable.js","../node_modules/rxjs/dist/esm5/internal/scheduled/scheduleArray.js","../node_modules/rxjs/dist/esm5/internal/scheduled/schedulePromise.js","../node_modules/rxjs/dist/esm5/internal/scheduled/scheduleIterable.js","../node_modules/rxjs/dist/esm5/internal/scheduled/scheduleReadableStreamLike.js","../node_modules/rxjs/dist/esm5/internal/observable/from.js","../node_modules/rxjs/dist/esm5/internal/observable/of.js","../node_modules/rxjs/dist/esm5/internal/operators/map.js","../node_modules/rxjs/dist/esm5/internal/util/mapOneOrManyArgs.js","../node_modules/rxjs/dist/esm5/internal/util/argsArgArrayOrObject.js","../node_modules/rxjs/dist/esm5/internal/observable/combineLatest.js","../node_modules/rxjs/dist/esm5/internal/util/createObject.js","../node_modules/rxjs/dist/esm5/internal/operators/mergeMap.js","../node_modules/rxjs/dist/esm5/internal/operators/mergeInternals.js","../node_modules/rxjs/dist/esm5/internal/operators/mergeAll.js","../node_modules/rxjs/dist/esm5/internal/observable/concat.js","../node_modules/rxjs/dist/esm5/internal/operators/concatAll.js","../node_modules/rxjs/dist/esm5/internal/observable/defer.js","../node_modules/rxjs/dist/esm5/internal/observable/timer.js","../node_modules/rxjs/dist/esm5/internal/util/isDate.js","../node_modules/rxjs/dist/esm5/internal/observable/merge.js","../node_modules/rxjs/dist/esm5/internal/operators/filter.js","../node_modules/rxjs/dist/esm5/internal/operators/take.js","../node_modules/rxjs/dist/esm5/internal/operators/distinctUntilChanged.js","../node_modules/rxjs/dist/esm5/internal/operators/endWith.js","../node_modules/rxjs/dist/esm5/internal/operators/finalize.js","../node_modules/rxjs/dist/esm5/internal/operators/merge.js","../node_modules/rxjs/dist/esm5/internal/operators/mergeWith.js","../node_modules/rxjs/dist/esm5/internal/operators/scan.js","../node_modules/rxjs/dist/esm5/internal/operators/scanInternals.js","../node_modules/rxjs/dist/esm5/internal/operators/share.js","../node_modules/rxjs/dist/esm5/internal/operators/switchMap.js","../node_modules/rxjs/dist/esm5/internal/operators/tap.js","../node_modules/applesauce-core/dist/helpers/tags.js","../node_modules/nostr-tools/lib/esm/pure.js","../node_modules/applesauce-core/dist/helpers/cache.js","../node_modules/nostr-tools/lib/esm/utils.js","../node_modules/applesauce-core/dist/helpers/event.js","../node_modules/applesauce-core/dist/helpers/time.js","../node_modules/applesauce-core/dist/helpers/expiration.js","../node_modules/nostr-tools/lib/esm/nip19.js","../node_modules/applesauce-core/dist/helpers/url.js","../node_modules/applesauce-core/dist/helpers/relays.js","../node_modules/applesauce-core/dist/helpers/string.js","../node_modules/@noble/hashes/esm/crypto.js","../node_modules/@noble/hashes/esm/utils.js","../node_modules/applesauce-core/dist/helpers/pointers.js","../node_modules/applesauce-core/dist/helpers/groups.js","../node_modules/applesauce-core/dist/helpers/encrypted-content.js","../node_modules/applesauce-core/dist/helpers/hidden-content.js","../node_modules/applesauce-core/dist/helpers/hidden-tags.js","../node_modules/applesauce-core/dist/helpers/event-tags.js","../node_modules/applesauce-core/dist/helpers/lru.js","../node_modules/ms/index.js","../node_modules/debug/src/common.js","../node_modules/debug/src/browser.js","../node_modules/applesauce-core/dist/logger.js","../node_modules/applesauce-core/dist/event-store/event-memory.js","../node_modules/hash-sum/hash-sum.js","../node_modules/applesauce-core/dist/helpers/filter.js","../node_modules/applesauce-core/dist/models/blossom.js","../node_modules/applesauce-core/dist/helpers/blossom.js","../node_modules/applesauce-core/dist/helpers/profile.js","../node_modules/applesauce-core/dist/helpers/json.js","../node_modules/light-bolt11-decoder/node_modules/@scure/base/lib/index.js","../node_modules/light-bolt11-decoder/bolt11.js","../node_modules/applesauce-core/dist/helpers/comment.js","../node_modules/applesauce-core/dist/helpers/contacts.js","../node_modules/applesauce-core/dist/helpers/encrypted-content-cache.js","../node_modules/applesauce-core/dist/helpers/event-cache.js","../node_modules/applesauce-core/dist/helpers/lnurl.js","../node_modules/applesauce-core/dist/helpers/mailboxes.js","../node_modules/applesauce-core/dist/helpers/mutes.js","../node_modules/applesauce-core/dist/helpers/reports.js","../node_modules/applesauce-core/dist/helpers/threading.js","../node_modules/applesauce-core/dist/observable/claim-events.js","../node_modules/applesauce-core/dist/observable/claim-latest.js","../node_modules/applesauce-core/dist/observable/defined.js","../node_modules/applesauce-core/dist/observable/with-immediate-value.js","../node_modules/applesauce-core/dist/models/common.js","../node_modules/rxjs/dist/esm5/internal/operators/ignoreElements.js","../node_modules/rxjs/dist/esm5/internal/operators/takeUntil.js","../node_modules/rxjs/dist/esm5/internal/operators/repeat.js","../node_modules/applesauce-core/dist/observable/watch-event-updates.js","../node_modules/applesauce-core/dist/models/contacts.js","../node_modules/applesauce-core/dist/models/comments.js","../node_modules/applesauce-core/dist/models/mailboxes.js","../node_modules/applesauce-core/dist/models/mutes.js","../node_modules/applesauce-core/dist/models/profile.js","../node_modules/applesauce-core/dist/models/reactions.js","../node_modules/applesauce-core/dist/models/thread.js","../node_modules/applesauce-core/dist/event-store/model-mixin.js","../node_modules/applesauce-core/dist/event-store/event-store.js","../node_modules/applesauce-core/dist/helpers/delete.js","../node_modules/applesauce-core/dist/promise/deferred.js","../node_modules/applesauce-signers/dist/signers/amber-clipboard-signer.js","../node_modules/applesauce-signers/dist/helpers/nostr-connect.js","../node_modules/applesauce-signers/dist/signers/private-key-signer.js","../node_modules/applesauce-signers/node_modules/@noble/secp256k1/lib/esm/index.js","../node_modules/applesauce-signers/node_modules/@scure/base/lib/esm/index.js","../node_modules/applesauce-signers/dist/signers/serial-port-signer.js","../src/constants.js","../src/nostr.js","../node_modules/hash-wasm/dist/index.esm.js","../src/nsec-crypto.js","../src/LoginModal.svelte","../src/ManagedACL.svelte","../src/Header.svelte","../src/Sidebar.svelte","../src/ExportView.svelte","../src/ImportView.svelte","../src/helpers.tsx","../src/FilterBuilder.svelte","../src/EventsView.svelte","../src/eventKinds.js","../src/EventTemplateSelector.svelte","../src/ComposeView.svelte","../src/SprocketView.svelte","../src/PolicyView.svelte","../src/kindCategories.js","../src/CurationView.svelte","../src/BlossomView.svelte","../src/LogView.svelte","../src/api.js","../src/utils.js","../src/RelayConnectView.svelte","../src/FilterDisplay.svelte","../src/websocket-auth.js","../src/App.svelte","../src/main.js"],"sourcesContent":["function noop() { }\nconst identity = x => x;\nfunction assign(tar, src) {\n // @ts-ignore\n for (const k in src)\n tar[k] = src[k];\n return tar;\n}\n// Adapted from https://github.com/then/is-promise/blob/master/index.js\n// Distributed under MIT License https://github.com/then/is-promise/blob/master/LICENSE\nfunction is_promise(value) {\n return !!value && (typeof value === 'object' || typeof value === 'function') && typeof value.then === 'function';\n}\nfunction add_location(element, file, line, column, char) {\n element.__svelte_meta = {\n loc: { file, line, column, char }\n };\n}\nfunction run(fn) {\n return fn();\n}\nfunction blank_object() {\n return Object.create(null);\n}\nfunction run_all(fns) {\n fns.forEach(run);\n}\nfunction is_function(thing) {\n return typeof thing === 'function';\n}\nfunction safe_not_equal(a, b) {\n return a != a ? b == b : a !== b || ((a && typeof a === 'object') || typeof a === 'function');\n}\nlet src_url_equal_anchor;\nfunction src_url_equal(element_src, url) {\n if (!src_url_equal_anchor) {\n src_url_equal_anchor = document.createElement('a');\n }\n src_url_equal_anchor.href = url;\n return element_src === src_url_equal_anchor.href;\n}\nfunction not_equal(a, b) {\n return a != a ? b == b : a !== b;\n}\nfunction is_empty(obj) {\n return Object.keys(obj).length === 0;\n}\nfunction validate_store(store, name) {\n if (store != null && typeof store.subscribe !== 'function') {\n throw new Error(`'${name}' is not a store with a 'subscribe' method`);\n }\n}\nfunction subscribe(store, ...callbacks) {\n if (store == null) {\n return noop;\n }\n const unsub = store.subscribe(...callbacks);\n return unsub.unsubscribe ? () => unsub.unsubscribe() : unsub;\n}\nfunction get_store_value(store) {\n let value;\n subscribe(store, _ => value = _)();\n return value;\n}\nfunction component_subscribe(component, store, callback) {\n component.$$.on_destroy.push(subscribe(store, callback));\n}\nfunction create_slot(definition, ctx, $$scope, fn) {\n if (definition) {\n const slot_ctx = get_slot_context(definition, ctx, $$scope, fn);\n return definition[0](slot_ctx);\n }\n}\nfunction get_slot_context(definition, ctx, $$scope, fn) {\n return definition[1] && fn\n ? assign($$scope.ctx.slice(), definition[1](fn(ctx)))\n : $$scope.ctx;\n}\nfunction get_slot_changes(definition, $$scope, dirty, fn) {\n if (definition[2] && fn) {\n const lets = definition[2](fn(dirty));\n if ($$scope.dirty === undefined) {\n return lets;\n }\n if (typeof lets === 'object') {\n const merged = [];\n const len = Math.max($$scope.dirty.length, lets.length);\n for (let i = 0; i < len; i += 1) {\n merged[i] = $$scope.dirty[i] | lets[i];\n }\n return merged;\n }\n return $$scope.dirty | lets;\n }\n return $$scope.dirty;\n}\nfunction update_slot_base(slot, slot_definition, ctx, $$scope, slot_changes, get_slot_context_fn) {\n if (slot_changes) {\n const slot_context = get_slot_context(slot_definition, ctx, $$scope, get_slot_context_fn);\n slot.p(slot_context, slot_changes);\n }\n}\nfunction update_slot(slot, slot_definition, ctx, $$scope, dirty, get_slot_changes_fn, get_slot_context_fn) {\n const slot_changes = get_slot_changes(slot_definition, $$scope, dirty, get_slot_changes_fn);\n update_slot_base(slot, slot_definition, ctx, $$scope, slot_changes, get_slot_context_fn);\n}\nfunction get_all_dirty_from_scope($$scope) {\n if ($$scope.ctx.length > 32) {\n const dirty = [];\n const length = $$scope.ctx.length / 32;\n for (let i = 0; i < length; i++) {\n dirty[i] = -1;\n }\n return dirty;\n }\n return -1;\n}\nfunction exclude_internal_props(props) {\n const result = {};\n for (const k in props)\n if (k[0] !== '$')\n result[k] = props[k];\n return result;\n}\nfunction compute_rest_props(props, keys) {\n const rest = {};\n keys = new Set(keys);\n for (const k in props)\n if (!keys.has(k) && k[0] !== '$')\n rest[k] = props[k];\n return rest;\n}\nfunction compute_slots(slots) {\n const result = {};\n for (const key in slots) {\n result[key] = true;\n }\n return result;\n}\nfunction once(fn) {\n let ran = false;\n return function (...args) {\n if (ran)\n return;\n ran = true;\n fn.call(this, ...args);\n };\n}\nfunction null_to_empty(value) {\n return value == null ? '' : value;\n}\nfunction set_store_value(store, ret, value) {\n store.set(value);\n return ret;\n}\nconst has_prop = (obj, prop) => Object.prototype.hasOwnProperty.call(obj, prop);\nfunction action_destroyer(action_result) {\n return action_result && is_function(action_result.destroy) ? action_result.destroy : noop;\n}\nfunction split_css_unit(value) {\n const split = typeof value === 'string' && value.match(/^\\s*(-?[\\d.]+)([^\\s]*)\\s*$/);\n return split ? [parseFloat(split[1]), split[2] || 'px'] : [value, 'px'];\n}\nconst contenteditable_truthy_values = ['', true, 1, 'true', 'contenteditable'];\n\nconst is_client = typeof window !== 'undefined';\nlet now = is_client\n ? () => window.performance.now()\n : () => Date.now();\nlet raf = is_client ? cb => requestAnimationFrame(cb) : noop;\n// used internally for testing\nfunction set_now(fn) {\n now = fn;\n}\nfunction set_raf(fn) {\n raf = fn;\n}\n\nconst tasks = new Set();\nfunction run_tasks(now) {\n tasks.forEach(task => {\n if (!task.c(now)) {\n tasks.delete(task);\n task.f();\n }\n });\n if (tasks.size !== 0)\n raf(run_tasks);\n}\n/**\n * For testing purposes only!\n */\nfunction clear_loops() {\n tasks.clear();\n}\n/**\n * Creates a new task that runs on each raf frame\n * until it returns a falsy value or is aborted\n */\nfunction loop(callback) {\n let task;\n if (tasks.size === 0)\n raf(run_tasks);\n return {\n promise: new Promise(fulfill => {\n tasks.add(task = { c: callback, f: fulfill });\n }),\n abort() {\n tasks.delete(task);\n }\n };\n}\n\nconst globals = (typeof window !== 'undefined'\n ? window\n : typeof globalThis !== 'undefined'\n ? globalThis\n : global);\n\n/**\n * Resize observer singleton.\n * One listener per element only!\n * https://groups.google.com/a/chromium.org/g/blink-dev/c/z6ienONUb5A/m/F5-VcUZtBAAJ\n */\nclass ResizeObserverSingleton {\n constructor(options) {\n this.options = options;\n this._listeners = 'WeakMap' in globals ? new WeakMap() : undefined;\n }\n observe(element, listener) {\n this._listeners.set(element, listener);\n this._getObserver().observe(element, this.options);\n return () => {\n this._listeners.delete(element);\n this._observer.unobserve(element); // this line can probably be removed\n };\n }\n _getObserver() {\n var _a;\n return (_a = this._observer) !== null && _a !== void 0 ? _a : (this._observer = new ResizeObserver((entries) => {\n var _a;\n for (const entry of entries) {\n ResizeObserverSingleton.entries.set(entry.target, entry);\n (_a = this._listeners.get(entry.target)) === null || _a === void 0 ? void 0 : _a(entry);\n }\n }));\n }\n}\n// Needs to be written like this to pass the tree-shake-test\nResizeObserverSingleton.entries = 'WeakMap' in globals ? new WeakMap() : undefined;\n\n// Track which nodes are claimed during hydration. Unclaimed nodes can then be removed from the DOM\n// at the end of hydration without touching the remaining nodes.\nlet is_hydrating = false;\nfunction start_hydrating() {\n is_hydrating = true;\n}\nfunction end_hydrating() {\n is_hydrating = false;\n}\nfunction upper_bound(low, high, key, value) {\n // Return first index of value larger than input value in the range [low, high)\n while (low < high) {\n const mid = low + ((high - low) >> 1);\n if (key(mid) <= value) {\n low = mid + 1;\n }\n else {\n high = mid;\n }\n }\n return low;\n}\nfunction init_hydrate(target) {\n if (target.hydrate_init)\n return;\n target.hydrate_init = true;\n // We know that all children have claim_order values since the unclaimed have been detached if target is not \n let children = target.childNodes;\n // If target is , there may be children without claim_order\n if (target.nodeName === 'HEAD') {\n const myChildren = [];\n for (let i = 0; i < children.length; i++) {\n const node = children[i];\n if (node.claim_order !== undefined) {\n myChildren.push(node);\n }\n }\n children = myChildren;\n }\n /*\n * Reorder claimed children optimally.\n * We can reorder claimed children optimally by finding the longest subsequence of\n * nodes that are already claimed in order and only moving the rest. The longest\n * subsequence of nodes that are claimed in order can be found by\n * computing the longest increasing subsequence of .claim_order values.\n *\n * This algorithm is optimal in generating the least amount of reorder operations\n * possible.\n *\n * Proof:\n * We know that, given a set of reordering operations, the nodes that do not move\n * always form an increasing subsequence, since they do not move among each other\n * meaning that they must be already ordered among each other. Thus, the maximal\n * set of nodes that do not move form a longest increasing subsequence.\n */\n // Compute longest increasing subsequence\n // m: subsequence length j => index k of smallest value that ends an increasing subsequence of length j\n const m = new Int32Array(children.length + 1);\n // Predecessor indices + 1\n const p = new Int32Array(children.length);\n m[0] = -1;\n let longest = 0;\n for (let i = 0; i < children.length; i++) {\n const current = children[i].claim_order;\n // Find the largest subsequence length such that it ends in a value less than our current value\n // upper_bound returns first greater value, so we subtract one\n // with fast path for when we are on the current longest subsequence\n const seqLen = ((longest > 0 && children[m[longest]].claim_order <= current) ? longest + 1 : upper_bound(1, longest, idx => children[m[idx]].claim_order, current)) - 1;\n p[i] = m[seqLen] + 1;\n const newLen = seqLen + 1;\n // We can guarantee that current is the smallest value. Otherwise, we would have generated a longer sequence.\n m[newLen] = i;\n longest = Math.max(newLen, longest);\n }\n // The longest increasing subsequence of nodes (initially reversed)\n const lis = [];\n // The rest of the nodes, nodes that will be moved\n const toMove = [];\n let last = children.length - 1;\n for (let cur = m[longest] + 1; cur != 0; cur = p[cur - 1]) {\n lis.push(children[cur - 1]);\n for (; last >= cur; last--) {\n toMove.push(children[last]);\n }\n last--;\n }\n for (; last >= 0; last--) {\n toMove.push(children[last]);\n }\n lis.reverse();\n // We sort the nodes being moved to guarantee that their insertion order matches the claim order\n toMove.sort((a, b) => a.claim_order - b.claim_order);\n // Finally, we move the nodes\n for (let i = 0, j = 0; i < toMove.length; i++) {\n while (j < lis.length && toMove[i].claim_order >= lis[j].claim_order) {\n j++;\n }\n const anchor = j < lis.length ? lis[j] : null;\n target.insertBefore(toMove[i], anchor);\n }\n}\nfunction append(target, node) {\n target.appendChild(node);\n}\nfunction append_styles(target, style_sheet_id, styles) {\n const append_styles_to = get_root_for_style(target);\n if (!append_styles_to.getElementById(style_sheet_id)) {\n const style = element('style');\n style.id = style_sheet_id;\n style.textContent = styles;\n append_stylesheet(append_styles_to, style);\n }\n}\nfunction get_root_for_style(node) {\n if (!node)\n return document;\n const root = node.getRootNode ? node.getRootNode() : node.ownerDocument;\n if (root && root.host) {\n return root;\n }\n return node.ownerDocument;\n}\nfunction append_empty_stylesheet(node) {\n const style_element = element('style');\n append_stylesheet(get_root_for_style(node), style_element);\n return style_element.sheet;\n}\nfunction append_stylesheet(node, style) {\n append(node.head || node, style);\n return style.sheet;\n}\nfunction append_hydration(target, node) {\n if (is_hydrating) {\n init_hydrate(target);\n if ((target.actual_end_child === undefined) || ((target.actual_end_child !== null) && (target.actual_end_child.parentNode !== target))) {\n target.actual_end_child = target.firstChild;\n }\n // Skip nodes of undefined ordering\n while ((target.actual_end_child !== null) && (target.actual_end_child.claim_order === undefined)) {\n target.actual_end_child = target.actual_end_child.nextSibling;\n }\n if (node !== target.actual_end_child) {\n // We only insert if the ordering of this node should be modified or the parent node is not target\n if (node.claim_order !== undefined || node.parentNode !== target) {\n target.insertBefore(node, target.actual_end_child);\n }\n }\n else {\n target.actual_end_child = node.nextSibling;\n }\n }\n else if (node.parentNode !== target || node.nextSibling !== null) {\n target.appendChild(node);\n }\n}\nfunction insert(target, node, anchor) {\n target.insertBefore(node, anchor || null);\n}\nfunction insert_hydration(target, node, anchor) {\n if (is_hydrating && !anchor) {\n append_hydration(target, node);\n }\n else if (node.parentNode !== target || node.nextSibling != anchor) {\n target.insertBefore(node, anchor || null);\n }\n}\nfunction detach(node) {\n if (node.parentNode) {\n node.parentNode.removeChild(node);\n }\n}\nfunction destroy_each(iterations, detaching) {\n for (let i = 0; i < iterations.length; i += 1) {\n if (iterations[i])\n iterations[i].d(detaching);\n }\n}\nfunction element(name) {\n return document.createElement(name);\n}\nfunction element_is(name, is) {\n return document.createElement(name, { is });\n}\nfunction object_without_properties(obj, exclude) {\n const target = {};\n for (const k in obj) {\n if (has_prop(obj, k)\n // @ts-ignore\n && exclude.indexOf(k) === -1) {\n // @ts-ignore\n target[k] = obj[k];\n }\n }\n return target;\n}\nfunction svg_element(name) {\n return document.createElementNS('http://www.w3.org/2000/svg', name);\n}\nfunction text(data) {\n return document.createTextNode(data);\n}\nfunction space() {\n return text(' ');\n}\nfunction empty() {\n return text('');\n}\nfunction comment(content) {\n return document.createComment(content);\n}\nfunction listen(node, event, handler, options) {\n node.addEventListener(event, handler, options);\n return () => node.removeEventListener(event, handler, options);\n}\nfunction prevent_default(fn) {\n return function (event) {\n event.preventDefault();\n // @ts-ignore\n return fn.call(this, event);\n };\n}\nfunction stop_propagation(fn) {\n return function (event) {\n event.stopPropagation();\n // @ts-ignore\n return fn.call(this, event);\n };\n}\nfunction stop_immediate_propagation(fn) {\n return function (event) {\n event.stopImmediatePropagation();\n // @ts-ignore\n return fn.call(this, event);\n };\n}\nfunction self(fn) {\n return function (event) {\n // @ts-ignore\n if (event.target === this)\n fn.call(this, event);\n };\n}\nfunction trusted(fn) {\n return function (event) {\n // @ts-ignore\n if (event.isTrusted)\n fn.call(this, event);\n };\n}\nfunction attr(node, attribute, value) {\n if (value == null)\n node.removeAttribute(attribute);\n else if (node.getAttribute(attribute) !== value)\n node.setAttribute(attribute, value);\n}\n/**\n * List of attributes that should always be set through the attr method,\n * because updating them through the property setter doesn't work reliably.\n * In the example of `width`/`height`, the problem is that the setter only\n * accepts numeric values, but the attribute can also be set to a string like `50%`.\n * If this list becomes too big, rethink this approach.\n */\nconst always_set_through_set_attribute = ['width', 'height'];\nfunction set_attributes(node, attributes) {\n // @ts-ignore\n const descriptors = Object.getOwnPropertyDescriptors(node.__proto__);\n for (const key in attributes) {\n if (attributes[key] == null) {\n node.removeAttribute(key);\n }\n else if (key === 'style') {\n node.style.cssText = attributes[key];\n }\n else if (key === '__value') {\n node.value = node[key] = attributes[key];\n }\n else if (descriptors[key] && descriptors[key].set && always_set_through_set_attribute.indexOf(key) === -1) {\n node[key] = attributes[key];\n }\n else {\n attr(node, key, attributes[key]);\n }\n }\n}\nfunction set_svg_attributes(node, attributes) {\n for (const key in attributes) {\n attr(node, key, attributes[key]);\n }\n}\nfunction set_custom_element_data_map(node, data_map) {\n Object.keys(data_map).forEach((key) => {\n set_custom_element_data(node, key, data_map[key]);\n });\n}\nfunction set_custom_element_data(node, prop, value) {\n if (prop in node) {\n node[prop] = typeof node[prop] === 'boolean' && value === '' ? true : value;\n }\n else {\n attr(node, prop, value);\n }\n}\nfunction set_dynamic_element_data(tag) {\n return (/-/.test(tag)) ? set_custom_element_data_map : set_attributes;\n}\nfunction xlink_attr(node, attribute, value) {\n node.setAttributeNS('http://www.w3.org/1999/xlink', attribute, value);\n}\nfunction get_binding_group_value(group, __value, checked) {\n const value = new Set();\n for (let i = 0; i < group.length; i += 1) {\n if (group[i].checked)\n value.add(group[i].__value);\n }\n if (!checked) {\n value.delete(__value);\n }\n return Array.from(value);\n}\nfunction init_binding_group(group) {\n let _inputs;\n return {\n /* push */ p(...inputs) {\n _inputs = inputs;\n _inputs.forEach(input => group.push(input));\n },\n /* remove */ r() {\n _inputs.forEach(input => group.splice(group.indexOf(input), 1));\n }\n };\n}\nfunction init_binding_group_dynamic(group, indexes) {\n let _group = get_binding_group(group);\n let _inputs;\n function get_binding_group(group) {\n for (let i = 0; i < indexes.length; i++) {\n group = group[indexes[i]] = group[indexes[i]] || [];\n }\n return group;\n }\n function push() {\n _inputs.forEach(input => _group.push(input));\n }\n function remove() {\n _inputs.forEach(input => _group.splice(_group.indexOf(input), 1));\n }\n return {\n /* update */ u(new_indexes) {\n indexes = new_indexes;\n const new_group = get_binding_group(group);\n if (new_group !== _group) {\n remove();\n _group = new_group;\n push();\n }\n },\n /* push */ p(...inputs) {\n _inputs = inputs;\n push();\n },\n /* remove */ r: remove\n };\n}\nfunction to_number(value) {\n return value === '' ? null : +value;\n}\nfunction time_ranges_to_array(ranges) {\n const array = [];\n for (let i = 0; i < ranges.length; i += 1) {\n array.push({ start: ranges.start(i), end: ranges.end(i) });\n }\n return array;\n}\nfunction children(element) {\n return Array.from(element.childNodes);\n}\nfunction init_claim_info(nodes) {\n if (nodes.claim_info === undefined) {\n nodes.claim_info = { last_index: 0, total_claimed: 0 };\n }\n}\nfunction claim_node(nodes, predicate, processNode, createNode, dontUpdateLastIndex = false) {\n // Try to find nodes in an order such that we lengthen the longest increasing subsequence\n init_claim_info(nodes);\n const resultNode = (() => {\n // We first try to find an element after the previous one\n for (let i = nodes.claim_info.last_index; i < nodes.length; i++) {\n const node = nodes[i];\n if (predicate(node)) {\n const replacement = processNode(node);\n if (replacement === undefined) {\n nodes.splice(i, 1);\n }\n else {\n nodes[i] = replacement;\n }\n if (!dontUpdateLastIndex) {\n nodes.claim_info.last_index = i;\n }\n return node;\n }\n }\n // Otherwise, we try to find one before\n // We iterate in reverse so that we don't go too far back\n for (let i = nodes.claim_info.last_index - 1; i >= 0; i--) {\n const node = nodes[i];\n if (predicate(node)) {\n const replacement = processNode(node);\n if (replacement === undefined) {\n nodes.splice(i, 1);\n }\n else {\n nodes[i] = replacement;\n }\n if (!dontUpdateLastIndex) {\n nodes.claim_info.last_index = i;\n }\n else if (replacement === undefined) {\n // Since we spliced before the last_index, we decrease it\n nodes.claim_info.last_index--;\n }\n return node;\n }\n }\n // If we can't find any matching node, we create a new one\n return createNode();\n })();\n resultNode.claim_order = nodes.claim_info.total_claimed;\n nodes.claim_info.total_claimed += 1;\n return resultNode;\n}\nfunction claim_element_base(nodes, name, attributes, create_element) {\n return claim_node(nodes, (node) => node.nodeName === name, (node) => {\n const remove = [];\n for (let j = 0; j < node.attributes.length; j++) {\n const attribute = node.attributes[j];\n if (!attributes[attribute.name]) {\n remove.push(attribute.name);\n }\n }\n remove.forEach(v => node.removeAttribute(v));\n return undefined;\n }, () => create_element(name));\n}\nfunction claim_element(nodes, name, attributes) {\n return claim_element_base(nodes, name, attributes, element);\n}\nfunction claim_svg_element(nodes, name, attributes) {\n return claim_element_base(nodes, name, attributes, svg_element);\n}\nfunction claim_text(nodes, data) {\n return claim_node(nodes, (node) => node.nodeType === 3, (node) => {\n const dataStr = '' + data;\n if (node.data.startsWith(dataStr)) {\n if (node.data.length !== dataStr.length) {\n return node.splitText(dataStr.length);\n }\n }\n else {\n node.data = dataStr;\n }\n }, () => text(data), true // Text nodes should not update last index since it is likely not worth it to eliminate an increasing subsequence of actual elements\n );\n}\nfunction claim_space(nodes) {\n return claim_text(nodes, ' ');\n}\nfunction claim_comment(nodes, data) {\n return claim_node(nodes, (node) => node.nodeType === 8, (node) => {\n node.data = '' + data;\n return undefined;\n }, () => comment(data), true);\n}\nfunction find_comment(nodes, text, start) {\n for (let i = start; i < nodes.length; i += 1) {\n const node = nodes[i];\n if (node.nodeType === 8 /* comment node */ && node.textContent.trim() === text) {\n return i;\n }\n }\n return nodes.length;\n}\nfunction claim_html_tag(nodes, is_svg) {\n // find html opening tag\n const start_index = find_comment(nodes, 'HTML_TAG_START', 0);\n const end_index = find_comment(nodes, 'HTML_TAG_END', start_index);\n if (start_index === end_index) {\n return new HtmlTagHydration(undefined, is_svg);\n }\n init_claim_info(nodes);\n const html_tag_nodes = nodes.splice(start_index, end_index - start_index + 1);\n detach(html_tag_nodes[0]);\n detach(html_tag_nodes[html_tag_nodes.length - 1]);\n const claimed_nodes = html_tag_nodes.slice(1, html_tag_nodes.length - 1);\n for (const n of claimed_nodes) {\n n.claim_order = nodes.claim_info.total_claimed;\n nodes.claim_info.total_claimed += 1;\n }\n return new HtmlTagHydration(claimed_nodes, is_svg);\n}\nfunction set_data(text, data) {\n data = '' + data;\n if (text.data === data)\n return;\n text.data = data;\n}\nfunction set_data_contenteditable(text, data) {\n data = '' + data;\n if (text.wholeText === data)\n return;\n text.data = data;\n}\nfunction set_data_maybe_contenteditable(text, data, attr_value) {\n if (~contenteditable_truthy_values.indexOf(attr_value)) {\n set_data_contenteditable(text, data);\n }\n else {\n set_data(text, data);\n }\n}\nfunction set_input_value(input, value) {\n input.value = value == null ? '' : value;\n}\nfunction set_input_type(input, type) {\n try {\n input.type = type;\n }\n catch (e) {\n // do nothing\n }\n}\nfunction set_style(node, key, value, important) {\n if (value == null) {\n node.style.removeProperty(key);\n }\n else {\n node.style.setProperty(key, value, important ? 'important' : '');\n }\n}\nfunction select_option(select, value, mounting) {\n for (let i = 0; i < select.options.length; i += 1) {\n const option = select.options[i];\n if (option.__value === value) {\n option.selected = true;\n return;\n }\n }\n if (!mounting || value !== undefined) {\n select.selectedIndex = -1; // no option should be selected\n }\n}\nfunction select_options(select, value) {\n for (let i = 0; i < select.options.length; i += 1) {\n const option = select.options[i];\n option.selected = ~value.indexOf(option.__value);\n }\n}\nfunction select_value(select) {\n const selected_option = select.querySelector(':checked');\n return selected_option && selected_option.__value;\n}\nfunction select_multiple_value(select) {\n return [].map.call(select.querySelectorAll(':checked'), option => option.__value);\n}\n// unfortunately this can't be a constant as that wouldn't be tree-shakeable\n// so we cache the result instead\nlet crossorigin;\nfunction is_crossorigin() {\n if (crossorigin === undefined) {\n crossorigin = false;\n try {\n if (typeof window !== 'undefined' && window.parent) {\n void window.parent.document;\n }\n }\n catch (error) {\n crossorigin = true;\n }\n }\n return crossorigin;\n}\nfunction add_iframe_resize_listener(node, fn) {\n const computed_style = getComputedStyle(node);\n if (computed_style.position === 'static') {\n node.style.position = 'relative';\n }\n const iframe = element('iframe');\n iframe.setAttribute('style', 'display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; ' +\n 'overflow: hidden; border: 0; opacity: 0; pointer-events: none; z-index: -1;');\n iframe.setAttribute('aria-hidden', 'true');\n iframe.tabIndex = -1;\n const crossorigin = is_crossorigin();\n let unsubscribe;\n if (crossorigin) {\n iframe.src = \"data:text/html,\";\n unsubscribe = listen(window, 'message', (event) => {\n if (event.source === iframe.contentWindow)\n fn();\n });\n }\n else {\n iframe.src = 'about:blank';\n iframe.onload = () => {\n unsubscribe = listen(iframe.contentWindow, 'resize', fn);\n // make sure an initial resize event is fired _after_ the iframe is loaded (which is asynchronous)\n // see https://github.com/sveltejs/svelte/issues/4233\n fn();\n };\n }\n append(node, iframe);\n return () => {\n if (crossorigin) {\n unsubscribe();\n }\n else if (unsubscribe && iframe.contentWindow) {\n unsubscribe();\n }\n detach(iframe);\n };\n}\nconst resize_observer_content_box = /* @__PURE__ */ new ResizeObserverSingleton({ box: 'content-box' });\nconst resize_observer_border_box = /* @__PURE__ */ new ResizeObserverSingleton({ box: 'border-box' });\nconst resize_observer_device_pixel_content_box = /* @__PURE__ */ new ResizeObserverSingleton({ box: 'device-pixel-content-box' });\nfunction toggle_class(element, name, toggle) {\n element.classList[toggle ? 'add' : 'remove'](name);\n}\nfunction custom_event(type, detail, { bubbles = false, cancelable = false } = {}) {\n const e = document.createEvent('CustomEvent');\n e.initCustomEvent(type, bubbles, cancelable, detail);\n return e;\n}\nfunction query_selector_all(selector, parent = document.body) {\n return Array.from(parent.querySelectorAll(selector));\n}\nfunction head_selector(nodeId, head) {\n const result = [];\n let started = 0;\n for (const node of head.childNodes) {\n if (node.nodeType === 8 /* comment node */) {\n const comment = node.textContent.trim();\n if (comment === `HEAD_${nodeId}_END`) {\n started -= 1;\n result.push(node);\n }\n else if (comment === `HEAD_${nodeId}_START`) {\n started += 1;\n result.push(node);\n }\n }\n else if (started > 0) {\n result.push(node);\n }\n }\n return result;\n}\nclass HtmlTag {\n constructor(is_svg = false) {\n this.is_svg = false;\n this.is_svg = is_svg;\n this.e = this.n = null;\n }\n c(html) {\n this.h(html);\n }\n m(html, target, anchor = null) {\n if (!this.e) {\n if (this.is_svg)\n this.e = svg_element(target.nodeName);\n /** #7364 target for