Compare commits
5 Commits
c0784fc890
...
f3068f82f3
| Author | SHA1 | Date | |
|---|---|---|---|
| f3068f82f3 | |||
|
|
a8dc2ed046 | ||
| 9a3965243c | |||
| 23c2a58c2b | |||
| 45fb6d061d |
10
POOL_API.md
10
POOL_API.md
@@ -16,7 +16,7 @@ This document describes the public API for the Nostr Relay Pool implementation i
|
||||
| [`nostr_relay_pool_poll()`](nostr_core/core_relay_pool.c:1232) | Single iteration poll and dispatch |
|
||||
| [`nostr_relay_pool_query_sync()`](nostr_core/core_relay_pool.c:695) | Synchronous query returning event array |
|
||||
| [`nostr_relay_pool_get_event()`](nostr_core/core_relay_pool.c:825) | Get single most recent event |
|
||||
| [`nostr_relay_pool_publish()`](nostr_core/core_relay_pool.c:866) | Publish event and wait for OK responses |
|
||||
| [`nostr_relay_pool_publish_async()`](nostr_core/core_relay_pool.c:866) | Publish event with async callbacks |
|
||||
| [`nostr_relay_pool_get_relay_status()`](nostr_core/core_relay_pool.c:944) | Get connection status for a relay |
|
||||
| [`nostr_relay_pool_list_relays()`](nostr_core/core_relay_pool.c:960) | List all relays and their statuses |
|
||||
| [`nostr_relay_pool_get_relay_stats()`](nostr_core/core_relay_pool.c:992) | Get detailed statistics for a relay |
|
||||
@@ -438,9 +438,9 @@ int get_latest_note_from_pubkey(nostr_relay_pool_t* pool, const char* pubkey_hex
|
||||
```
|
||||
|
||||
### Publish Event
|
||||
**Function:** [`nostr_relay_pool_publish()`](nostr_core/core_relay_pool.c:866)
|
||||
**Function:** [`nostr_relay_pool_publish_async()`](nostr_core/core_relay_pool.c:866)
|
||||
```c
|
||||
int nostr_relay_pool_publish(
|
||||
int nostr_relay_pool_publish_async(
|
||||
nostr_relay_pool_t* pool,
|
||||
const char** relay_urls,
|
||||
int relay_count,
|
||||
@@ -471,8 +471,8 @@ int publish_text_note(nostr_relay_pool_t* pool, const char* content) {
|
||||
|
||||
printf("Publishing note: %s\n", content);
|
||||
|
||||
int success_count = nostr_relay_pool_publish(
|
||||
pool, relay_urls, 3, event);
|
||||
int success_count = nostr_relay_pool_publish_async(
|
||||
pool, relay_urls, 3, event, my_callback, user_data);
|
||||
|
||||
cJSON_Delete(event);
|
||||
|
||||
|
||||
233
debug.log
233
debug.log
@@ -1052,3 +1052,236 @@
|
||||
[04:51:30.102] SEND relay.laantungir.net:443: ["CLOSE", "pool_1_1759481260"]
|
||||
[04:51:30.102] SEND relay.laantungir.net:443: ["CLOSE", "pool_2_1759481341"]
|
||||
[04:51:30.102] SEND nos.lol:443: ["CLOSE", "pool_2_1759481341"]
|
||||
|
||||
=== NOSTR WebSocket Debug Log Started ===
|
||||
[11:36:03.026] SEND relay.laantungir.net:443: ["EVENT", {
|
||||
"pubkey": "650b42b2d501cd03b490ee27205444d1e507c200c3a137b99710564317eea65d",
|
||||
"created_at": 1759678563,
|
||||
"kind": 1,
|
||||
"tags": [],
|
||||
"content": "Test post at 2025-10-05 11:36:03",
|
||||
"id": "2a77ce51810b77073a7873fd00c56babde944a2520a49f852121f2438155e282",
|
||||
"sig": "a933a5f072a3a122bb7c685f6810781aaec47493e62f3f2540f49d7a8e7506be5d3dc319d98781a1807f4837acbb5ae1b76d8f99ce92ea55553a2e6eeec2697a"
|
||||
}]
|
||||
[11:36:03.104] RECV relay.laantungir.net:443: ["OK","2a77ce51810b77073a7873fd00c56babde944a2520a49f852121f2438155e282",true,""]
|
||||
|
||||
=== NOSTR WebSocket Debug Log Started ===
|
||||
[14:00:19.331] SEND localhost:7555: ["EVENT", {
|
||||
"pubkey": "7c2065299249ff1b4316c07af0d03d67ad145da80b907cc22aa608218c6d2687",
|
||||
"created_at": 1759687219,
|
||||
"kind": 1,
|
||||
"tags": [],
|
||||
"content": "Test post at 2025-10-05 14:00:19",
|
||||
"id": "15dbe2cfe9237b93992a3b4de9d90224032e8b613031748da92c46c1aeaa2e9c",
|
||||
"sig": "c0aa610658d8cfd76aca25894276a7f3f0573c2993429feff8b1ff10eeb5608ff18cfd79126de5f54e77f5b45a9d39f9006df65b56e4246e50cf72d212509569"
|
||||
}]
|
||||
[14:00:19.334] RECV localhost:7555: ["OK", "15dbe2cfe9237b93992a3b4de9d90224032e8b613031748da92c46c1aeaa2e9c", true, ""]
|
||||
[14:00:49.419] SEND localhost:7555: ["EVENT", {
|
||||
"pubkey": "2a0c81450868960944c17b0f57f6aa2b5307b56bc8d9cc1b213674c561a3ab85",
|
||||
"created_at": 1759687249,
|
||||
"kind": 1,
|
||||
"tags": [],
|
||||
"content": "Test post at 2025-10-05 14:00:49",
|
||||
"id": "c70d6c5c8745c088bd3b99e5f529bb69cd3d162cb9db28aa433a798fa8d811c3",
|
||||
"sig": "76ac20012e84d49a528db140e48024d6a390033df75bfbad64e822208a6a4a3f863b84860da8bcf78c6d8c8933e30fba92f11ebca5f4602ba4428b10584c7ceb"
|
||||
}]
|
||||
[14:01:23.848] SEND localhost:7555: ["EVENT", {
|
||||
"pubkey": "aa3b44608a9e52473a631d596c395dd297443237d56e34e31bdcd4272a1cf558",
|
||||
"created_at": 1759687283,
|
||||
"kind": 1,
|
||||
"tags": [],
|
||||
"content": "Test post at 2025-10-05 14:01:23",
|
||||
"id": "ec98292f570049ed00ae541cba716dcf6c8252b523ce036d9d3d79946ff62927",
|
||||
"sig": "3e5ffa59e881c4e772e3c0e65b26d5396b7687de2ae8e930b505fc2617ecfa4b392df32b3af41f8c989e5cb1eeb00d5c1fbedfba2729eb43a3838e49ebfffc8f"
|
||||
}]
|
||||
[14:01:23.851] RECV localhost:7555: ["OK", "ec98292f570049ed00ae541cba716dcf6c8252b523ce036d9d3d79946ff62927", true, ""]
|
||||
[14:02:12.846] SEND localhost:7555: ["REQ", "pool_1_1759687332", {
|
||||
"kinds": [1],
|
||||
"since": 1759687324,
|
||||
"limit": 10
|
||||
}]
|
||||
[14:02:13.462] SEND nos.lol:443: ["REQ", "pool_1_1759687332", {
|
||||
"kinds": [1],
|
||||
"since": 1759687324,
|
||||
"limit": 10
|
||||
}]
|
||||
[14:02:13.462] SEND nos.lol:443: ["REQ", "pool_1_1759687332", {
|
||||
"kinds": [1],
|
||||
"since": 1759687324,
|
||||
"limit": 10
|
||||
}]
|
||||
[14:02:13.614] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"The current price of Bitcoin is ~$123135.80\n\nBlock Height : 917792\n\n[2025-10-05 18:02:03 UTC] \n\n #bitcoin #nxus #nyx #oracle","created_at":1759687324,"id":"dfd294e7e51a9d8d323a56061dc93402b01766e64bb8da0b68e564c780c3db8a","kind":1,"pubkey":"ff8464aa592cdec49f76bf3c0b505433c47427d04eeaec1fa4c4e104709dabc3","sig":"0309e8ff8a0f1d0d36f990fcdde78d2d9e9125229d1ab850fe92fbbabffe5f26ca5204a48c44bab0044aeeb1824b7dd54d9e9074bad5e3263691c445553ed2de","tags":[]}]
|
||||
[14:02:13.675] RECV nos.lol:443: ["EOSE","pool_1_1759687332"]
|
||||
[14:02:14.774] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"Why?","created_at":1759687333,"id":"c0c29a6a16ed277868962e6664c07a8be2afd416c8b0212c9f17eff1a073ac15","kind":1,"pubkey":"8a719d42268aa925f6b12bcbe6b645f6ec82efe64eac91daf935fdcccf1f0fe5","sig":"db6174f60d86859989e34f6963c0ad25bfe67002a35d47933ad3c7ce7b271d061d82a7ed702c0aa6419eafb83cc0c93675b675bdc180d5d859df8691c52de38d","tags":[["alt","A short note: Why?"],["e","c2d6913771cbbbdd3fefaf8f016a142841b5d00278bd16359c6e011ec1a33539","wss://relay.snort.social/","root","341db5a7e3a931f49095d82a4acc939cf8a67293b1e4179fd4b5c0544c4fc2ef"],["p","341db5a7e3a931f49095d82a4acc939cf8a67293b1e4179fd4b5c0544c4fc2ef","wss://nostr.bitcoiner.social/"]]}]
|
||||
[14:02:15.737] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"check out ⒷⒶⓃⒼⒻⓄⓇⒻⓊⓃ .ⒸⓄⓂ its filled with nothing but everyday girls who are lonely due to covid and want to exchange contact info with her again if you would be appreciated","created_at":1759687335,"id":"02982bdbc18d00777cd274270cee086b30847a93f875b5d0417135c406443156","kind":1,"pubkey":"fe5f59754d2e3a2579b4c95d67f7ff12da4bfffbf8f1c6b0a8de3970407ef974","sig":"cfa9ad66756503599e83a0ec77897cf0170b44a71ced1bf6e1843a146f1d0a381ffdf951e20d1f05814ffac549425435b4519a138ccec691f6d2a2e7648d19bf","tags":[]}]
|
||||
[14:02:22.358] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"Sunday\ndon’t forget to stretch \nhttps://blossom.primal.net/d18ce68340127803a368333762a18caf6083762b9bb29e6a48ca806d5fc89f92.mov","created_at":1759687342,"id":"97e92a02de8c446067ed5fef48307139eac31caa81bde9152c0bfe7b4b33f5c9","kind":1,"pubkey":"8fda2199c4399fad7ff4126b402f1f5ee157fdbe4c0951e64dfe31aaec877ff1","sig":"e599ca1b54e9fe930083a6c8d1a967be347656b5a45939ead9156cba7f3e89946613a259bb38816778eee22a5bffb9c825eb3555edd09f79304ce529774349b0","tags":[]}]
|
||||
[14:02:24.129] SEND localhost:7555: ["REQ", "pool_1_1759687332", {
|
||||
"kinds": [1],
|
||||
"since": 1759687324,
|
||||
"limit": 10
|
||||
}]
|
||||
[14:02:24.130] RECV localhost:7555: ["EOSE", "pool_1_1759687332"]
|
||||
[14:02:24.890] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"Indeed he is 😅🫡","created_at":1759687345,"id":"097a6a383c2fbc3f65dd37752ad941545a1295e786758392df0bf9a6b2327a30","kind":1,"pubkey":"db8f291dcf949373f5224070cbef4fab80b9a8d3434246aca1fe34114cc51dd0","sig":"2d0cf79a141c3bb06245bee49c601b5f415c601f8d40f2d444ccdf90e79ec6254d1dc46b729b136030f19775a65a5a4655d5f40e9954df0c894a1149ca3305ec","tags":[["p","493a5a07be1dc06a1dd2d8e943d8924fd469f057857e26f7b99ebdebe00c2557","","mention"],["p","db8f291dcf949373f5224070cbef4fab80b9a8d3434246aca1fe34114cc51dd0","","mention"],["p","496bf22b76e63553b2cac70c44b53867368b4b7612053a2c78609f3144324807","","mention"],["e","bd01a782c099d9a4121063df8d89e6722d4b54d3015624870b7c80acd047f739","","root"],["e","c0ac6ca2931e1e22ab99c3daf5f1c85be1644ab13848d5d136eda170b95e0911","","reply"]]}]
|
||||
[14:02:41.565] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"I never know when to stfu","created_at":1759687362,"id":"291797f5b41d6a22e41b7001ddb8f68c60175408f13633d686f6f738366fa304","kind":1,"pubkey":"341db5a7e3a931f49095d82a4acc939cf8a67293b1e4179fd4b5c0544c4fc2ef","sig":"003486d469652f3485106de99d3c156d23625df98748471b36ffe207a966ad414621d36dd36cb1206ae17dd41741cc4f047a3ce2d706abc296cde351752f6c24","tags":[["alt","A short note: I never know when to stfu"],["e","c2d6913771cbbbdd3fefaf8f016a142841b5d00278bd16359c6e011ec1a33539","wss://nostr.bitcoiner.social/","root","341db5a7e3a931f49095d82a4acc939cf8a67293b1e4179fd4b5c0544c4fc2ef"],["e","c0c29a6a16ed277868962e6664c07a8be2afd416c8b0212c9f17eff1a073ac15","wss://wot.nostr.net/","reply","8a719d42268aa925f6b12bcbe6b645f6ec82efe64eac91daf935fdcccf1f0fe5"],["p","341db5a7e3a931f49095d82a4acc939cf8a67293b1e4179fd4b5c0544c4fc2ef","wss://nostr.bitcoiner.social/"],["p","8a719d42268aa925f6b12bcbe6b645f6ec82efe64eac91daf935fdcccf1f0fe5","ws://agwwuih4l66mb6oxnqk42lfsubhatux3bmcnpmki6cd7nmx5lz2ys6yd.onion/"]]}]
|
||||
[14:02:42.621] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"歯を磨きました えらいです","created_at":1759687359,"id":"b0c6315ae6940ec80b90f3fd5afaa87bbc608e266da91a5d69a62455ccb1ada0","kind":1,"pubkey":"8c59239319637f97e007dad0d681e65ce35b1ace333b629e2d33f9465c132608","sig":"84cae19a484de2d28cf0458920ebf918af55c5c244796301ddc2c050adc6f8d177f6dd833a2bd3c2fe0cbde82a1c4ab5ac29fba3f7d88829b18d3717fbdcc4de","tags":[]}]
|
||||
[14:02:45.057] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"FREEDOM!!!🥴","created_at":1759687364,"id":"7bc750b643cda5fd7b38e4ff9c2bc085b7af1ea3710617d8fc13d5c84af74c41","kind":1,"pubkey":"b7baced2435c5a2427cadcefd9257786aaab6ab688b105f32e2657374da9e74c","sig":"816efcd39f190de6b21ced158e1c86611a1892ff2bf4b0a5f3ee22bce3fdd8f9488865aab28705527f19062d86f1848ecb94a107a63818c3c094c7c4a624cc10","tags":[["e","8536277b5d44700c12e4c1d2d3571f22d810bb90baa3559ae4cbba36e0f3b8dc","","root"],["p","526b4dba54b3f4a1bdac3548f0b8195e2b8ea36ba1920c801eb6388ed67de4ec"]]}]
|
||||
[14:02:46.930] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"GN https://blossom.primal.net/9e0dee7527fe3a20428bcfd20f1a3057f21991d0eb34b69b328a636a5a05a5b9.jpg","created_at":1759687367,"id":"e0415b2de12951a6b0b4274ad1c7898c6d4ff0ed3f5169dc2dd2329583aa6ac7","kind":1,"pubkey":"621889de151ed29c32424bfb150066dc274808dd8545898b62154e20595deab5","sig":"15821cd116d0086ce5e8d6ba7320eff13ccc27edaeba958e4e122d9b3123026aae17d2350e7f0ad4b1723b871e26b0b75980b3f0291a4a5ffc7306ed49c0a93e","tags":[["alt","A short note: GN https://blossom.primal.net/9e0dee7527fe3a20428b..."],["r","https://blossom.primal.net/9e0dee7527fe3a20428bcfd20f1a3057f21991d0eb34b69b328a636a5a05a5b9.jpg"],["imeta","url https://blossom.primal.net/9e0dee7527fe3a20428bcfd20f1a3057f21991d0eb34b69b328a636a5a05a5b9.jpg","x 9e0dee7527fe3a20428bcfd20f1a3057f21991d0eb34b69b328a636a5a05a5b9","size 1296120","m image/jpeg","dim 4032x3024","blurhash #nH2G--TWAR-WCoLoeofj@_NodjYfkayj[fPj[fQJqR+oes.oeazWCWCazE3WXj[j@j[azf6f6f7V@ofWVafayjtj[j[j@%0oeayf6f6j[bHbHa|t6ayj[j[j[a|ayaya|","ox 9e0dee7527fe3a20428bcfd20f1a3057f21991d0eb34b69b328a636a5a05a5b9","alt "]]}]
|
||||
[14:02:56.059] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"#hipwaist ratios \nhttps://images2.imgbox.com/b1/0c/wCxDWNqC_o.jpg \n#JoyOfLife ","created_at":1759687364,"id":"7c2f33b25929cc70c60f0165b5fde62753bdde14a5a9daad012073f7189919b5","kind":1,"pubkey":"ebb93aa6c2d04d7368ba52ef69884bf6e5779d7e418187e33371a20c7d9e5f6f","sig":"bb0c6ea5bd499637d454459a349aeddf10594af224ab12ded0fb3152ee17f53e7a75c6c784cabb2e8ad4d9c59c1f2bd1ed21f3ef31c674fd82f9a8d3e43e4cc2","tags":[]}]
|
||||
[14:03:03.608] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"ほろ酔いマスカットティーサワーを飲んでいます","created_at":1759687383,"id":"df1a4aa1e449f96295902ac73d6f85dbb10273a063a91de90406522cf78ae98b","kind":1,"pubkey":"e62f27d2814a25171c466d2d7612ad1a066db1362b4e259db5c076f9e6b21cb7","sig":"3a6e49bad5b31a5218465c9bf9fbe4dba46731fefc4ad09e85139d256f471904662397b9100aec8beedb3635324356ea25370c957543b7e47fb75817db70c1f6","tags":[]}]
|
||||
[14:03:07.944] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"🟠 BTC: $123,118\n⛏️ Hashrate: 1109744.2 EH/s\n📦 Mempool size: 38.97 MB\n💸 Fees (sat/vB): low 1, med 3, high 4\n📊 Fear & Greed Index: 74 (Greed)\n#bitcoin #stats","created_at":1759687387,"id":"2ffa7922bb8706b4961a56cea3a4df70c85b79f2a366d9add75e65a23ed44cfb","kind":1,"pubkey":"e185c2ad0b87b3207ac2b96d6b8fb1ff10fbf25f93eef4d04fb6dbb9039f19fb","sig":"fe684f31bc65d3a7f524d9ad2c437cab752c18f05c5332761f35cbb67894a62a4c58efc5c94dedf285357e4fdc3c7f31c2a93818e1aed9745983de272c56a1cc","tags":[]}]
|
||||
[14:03:11.336] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"Fees incentivize long-term security. #bitcoinfees | BTC: $123,109 | Educational Post","created_at":1759687390,"id":"f1d0cb6a9409534a33320905a34ab1569a46e3f2d5cf429f669002dea789ec38","kind":1,"pubkey":"e185c2ad0b87b3207ac2b96d6b8fb1ff10fbf25f93eef4d04fb6dbb9039f19fb","sig":"34c033fa98e7248e7b4f038fd0180ca52b9309be707d3b6d33c9dbc6234ec5abecfd5dffb5d3ec053c6e58829c74ccda6369bcabcc89cfaff278451bafc23525","tags":[]}]
|
||||
[14:03:15.656] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"📈 BTC Price Last 7 Days #bitcoin #chart\n\n(Chart could not be uploaded)","created_at":1759687395,"id":"aa169c771966a25c3b6342868d8f3d903b22c1c778cc44d8d3c26c8d5141b5e4","kind":1,"pubkey":"e185c2ad0b87b3207ac2b96d6b8fb1ff10fbf25f93eef4d04fb6dbb9039f19fb","sig":"057ad92d9597617646ebf4c6acc0b428ac3e3b95e6c1cb6789a61900f1c385513441a0b380bb43af8b683f93f33837f6a195c94e3905bed6ef3556fab0d68397","tags":[]}]
|
||||
[14:03:16.063] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"Anybody lurk on \"X\" using nitter?","created_at":1759687397,"id":"abc512ddf94c69c4311c5d3067d43d6359d679daaa2655fb80a228a9428b0272","kind":1,"pubkey":"fd31d0a4910f7ce9c46ac63df62781a9981e79a11de16d1664f1feb14f6a8c94","sig":"8abeb53756cea6e042172ffaba2158d824f62accc795bc1d9b4b360b5b7060ad45ec550a39fc456d7f19dce4c06a6b92be61199f10af3ecb8b0077c9cc3768a0","tags":[]}]
|
||||
[14:03:19.498] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"😳","created_at":1759687399,"id":"2bd27d8cc846e0fb6ef37df032f3a26cdab82c837c9ec5e5845b3c0df0b0fa84","kind":1,"pubkey":"8fda2199c4399fad7ff4126b402f1f5ee157fdbe4c0951e64dfe31aaec877ff1","sig":"dc584de809ae64fabb33f3e4ddc677f8e94c6c3639931e9e600cd83804b3d6085eaf3dd591b3d5ff4b20e08baeec30904c7cee156a30d92c9fe5f99ed621c246","tags":[["e","0cd929d752df6c6f50071054986bcd34e4618e2c24c9d943cf66dd3e5d7ccc3f","wss://relay.damus.io/","root","8fda2199c4399fad7ff4126b402f1f5ee157fdbe4c0951e64dfe31aaec877ff1"],["e","e27f1ec0adeba973bd0858b76e74836df2d68fc57fc003ec3fb0f1603280f660","","reply"],["p","cdecc31c6d9406e9a7d6b0067412aa661d9d31c8035c3fd65c06301d1cac3b92","","mention"]]}]
|
||||
[14:03:22.260] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"Looks like a great time, good luck. \n\n","created_at":1759687402,"id":"e9430e5efb1beb33963c112d0d92387d4165ae41d66e81439a2405630b1ae129","kind":1,"pubkey":"9ce3f065a7f384c1927263b968726d08e3895b5823be6e322545e89c3ee31a15","sig":"670788f0c36701cbfddfc101c64134b4eee98225bbf31d982afb2e90a617278ba84c00a9bb530ccded4d2e0bee347c9620298791bedb9894bfe379e3b82ecea0","tags":[["p","15652c499840af95bf52d8eb93bfbdd734b92d40f21c4eb974b124e5997d57cb","wss://relay.primal.net"],["e","a411057eb22fe7ceda41de50d34259f974ff434e370b157ac51850b4802d31ba","wss://relay.primal.net","root","15652c499840af95bf52d8eb93bfbdd734b92d40f21c4eb974b124e5997d57cb"]]}]
|
||||
[14:03:30.391] SEND localhost:7555: ["EVENT", {
|
||||
"pubkey": "17323141f3a98b014f96ee85df59ee1ce6295e1502644c8e1f86439b20908198",
|
||||
"created_at": 1759687410,
|
||||
"kind": 1,
|
||||
"tags": [],
|
||||
"content": "Test post at 2025-10-05 14:03:30",
|
||||
"id": "8433206a6e00ff021ddd1f162af38bf724aabd6a7ca5fb9e57bda74f26d46e3a",
|
||||
"sig": "058a3bc6872e44e8d30cfba1d75267929ec3d73fbafa4c77f99897b700020a42478b51ee3bf4a4901beb376b9d3deca2854b3bca2f18eab169f9f5babdfeccb6"
|
||||
}]
|
||||
[14:03:30.394] RECV localhost:7555: ["EVENT", "pool_1_1759687332", {
|
||||
"pubkey": "17323141f3a98b014f96ee85df59ee1ce6295e1502644c8e1f86439b20908198",
|
||||
"created_at": 1759687410,
|
||||
"kind": 1,
|
||||
"tags": [],
|
||||
"content": "Test post at 2025-10-05 14:03:30",
|
||||
"id": "8433206a6e00ff021ddd1f162af38bf724aabd6a7ca5fb9e57bda74f26d46e3a",
|
||||
"sig": "058a3bc6872e44e8d30cfba1d75267929ec3d73fbafa4c77f99897b700020a42478b51ee3bf4a4901beb376b9d3deca2854b3bca2f18eab169f9f5babdfeccb6"
|
||||
}]
|
||||
[14:03:30.394] RECV localhost:7555: EVENT", "pool_1_1759687332", {
|
||||
"pubkey": "17323141f3a98b014f96ee85df59ee1ce6295e1502644c8e1f86439b20908198",
|
||||
"created_at": 1759687410,
|
||||
"kind": 1,
|
||||
"tags": [],
|
||||
"content": "Test post at 2025-10-05 14:03:30",
|
||||
"id": "8433206a6e00ff021ddd1f162af38bf724aabd6a7ca5fb9e57bda74f26d46e3a",
|
||||
"sig": "058a3bc6872e44e8d30cfba1d75267929ec3d73fbafa4c77f99897b700020a42478b51ee3bf4a4901beb376b9d3deca2854b3bca2f18eab169f9f5babdfeccb6"
|
||||
}]2a
|
||||
[14:03:30.746] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"You will eventually","created_at":1759687410,"id":"f8bad59298e8afa0cceb9a4dee73c28648b62846b1fedc13b450316b29e3c460","kind":1,"pubkey":"8a719d42268aa925f6b12bcbe6b645f6ec82efe64eac91daf935fdcccf1f0fe5","sig":"4313cc6ee55b482e67f9a1f2e33761b9c68158b59eee38419d540ce5bb6564f2d5982428e0cde14c260edc4354a9c2cd3f0b848f7dca25af8f5b8930b9577f1e","tags":[["alt","A short note: You will eventually"],["e","c2d6913771cbbbdd3fefaf8f016a142841b5d00278bd16359c6e011ec1a33539","wss://relay.snort.social/","root","341db5a7e3a931f49095d82a4acc939cf8a67293b1e4179fd4b5c0544c4fc2ef"],["e","c0c29a6a16ed277868962e6664c07a8be2afd416c8b0212c9f17eff1a073ac15","wss://sendit.nosflare.com/","","8a719d42268aa925f6b12bcbe6b645f6ec82efe64eac91daf935fdcccf1f0fe5"],["e","291797f5b41d6a22e41b7001ddb8f68c60175408f13633d686f6f738366fa304","wss://nostr.mom/","reply","341db5a7e3a931f49095d82a4acc939cf8a67293b1e4179fd4b5c0544c4fc2ef"],["p","341db5a7e3a931f49095d82a4acc939cf8a67293b1e4179fd4b5c0544c4fc2ef","wss://nostr.bitcoiner.social/"],["p","8a719d42268aa925f6b12bcbe6b645f6ec82efe64eac91daf935fdcccf1f0fe5","ws://agwwuih4l66mb6oxnqk42lfsubhatux3bmcnpmki6cd7nmx5lz2ys6yd.onion/"]]}]
|
||||
[14:03:34.291] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"お茶系のお酒あんまり得意じゃないんだけど、お茶って感じあんまりしないし大丈夫だった","created_at":1759687413,"id":"790a85a90ef5d2bfbd9c70fc1c54830333066477c63e0026ae304037e6272814","kind":1,"pubkey":"e62f27d2814a25171c466d2d7612ad1a066db1362b4e259db5c076f9e6b21cb7","sig":"47eea4b740aeba09a6ec595dd7fcd2d7a682d8091301a9cef3c014693ca639679eaabbca981030bbb23c7f683a9ee1faa7c18fc78a254d6b965b1723f77c8172","tags":[]}]
|
||||
[14:03:35.459] RECV nos.lol:443: ["EVENT","pool_1_1759687332",{"content":"No, this is Albuquerque, NM ","created_at":1759687415,"id":"edc6b801a73667504bb197b8315f9b4c0533f726c108f8932d3fdc31a6495553","kind":1,"pubkey":"da26e54b86c9a395a4233cbb540fe2aa93cdad4a9b657ed5a724efed5859d23d","sig":"b6b822662302cc6b21b09b6e50f3ecf45c09c90c3750b308e6e293f5037352bacd8e4f96706af43dfe584d9c6c295dc7a92420ab4262e73759af71f35e2bf25e","tags":[["e","021147fc341540353e9cf639b7d6d1ba28af3ece5cf0625484716b0228a9348d","wss://feeds.nostr.band/lang/en","root"],["e","0dc0f9c4a0f5f37d93742e7dcf7c0f9b72f9be48dd5503f49004975b0073c235","wss://relay.primal.net","reply"],["p","c1503e4a2ef14908127d373530480c133bd5a0d9bb47a4be4a25a41d8f27447c","","mention"]]}]
|
||||
[14:03:40.010] SEND nos.lol:443: ["EVENT", {
|
||||
"pubkey": "17323141f3a98b014f96ee85df59ee1ce6295e1502644c8e1f86439b20908198",
|
||||
"created_at": 1759687410,
|
||||
"kind": 1,
|
||||
"tags": [],
|
||||
"content": "Test post at 2025-10-05 14:03:30",
|
||||
"id": "8433206a6e00ff021ddd1f162af38bf724aabd6a7ca5fb9e57bda74f26d46e3a",
|
||||
"sig": "058a3bc6872e44e8d30cfba1d75267929ec3d73fbafa4c77f99897b700020a42478b51ee3bf4a4901beb376b9d3deca2854b3bca2f18eab169f9f5babdfeccb6"
|
||||
}]
|
||||
[14:03:40.207] RECV nos.lol:443: ["OK","8433206a6e00ff021ddd1f162af38bf724aabd6a7ca5fb9e57bda74f26d46e3a",true,""]
|
||||
[14:03:50.096] SEND localhost:7555: ["REQ", "pool_1_1759687332", {
|
||||
"kinds": [1],
|
||||
"since": 1759687324,
|
||||
"limit": 10
|
||||
}]
|
||||
[14:03:50.097] RECV localhost:7555: ["EVENT", "pool_1_1759687332", {
|
||||
"id": "8433206a6e00ff021ddd1f162af38bf724aabd6a7ca5fb9e57bda74f26d46e3a",
|
||||
"pubkey": "17323141f3a98b014f96ee85df59ee1ce6295e1502644c8e1f86439b20908198",
|
||||
"created_at": 1759687410,
|
||||
"kind": 1,
|
||||
"content": "Test post at 2025-10-05 14:03:30",
|
||||
"sig": "058a3bc6872e44e8d30cfba1d75267929ec3d73fbafa4c77f99897b700020a42478b51ee3bf4a4901beb376b9d3deca2854b3bca2f18eab169f9f5babdfeccb6",
|
||||
"tags": []
|
||||
}]
|
||||
[14:03:50.108] RECV localhost:7555: ["EOSE", "pool_1_1759687332"]
|
||||
|
||||
=== NOSTR WebSocket Debug Log Started ===
|
||||
[18:53:08.601] SEND localhost:7555: ["REQ", "pool_1_1759791188", {
|
||||
"kinds": [1],
|
||||
"since": 1759791185,
|
||||
"limit": 10
|
||||
}]
|
||||
[18:53:08.601] SEND nos.lol:443: ["REQ", "pool_1_1759791188", {
|
||||
"kinds": [1],
|
||||
"since": 1759791185,
|
||||
"limit": 10
|
||||
}]
|
||||
[18:53:08.648] RECV localhost:7555: ["EOSE", "pool_1_1759791188"]
|
||||
[18:53:08.783] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"🟩BUY BTC with USD\nPrice: 123770.37USD (-1%)\nBTC: 0.005 - 0.01\nUSD: 619 - 1238\nMethod: ACH\nCreated: 2025-10-06T22:52:29Z","created_at":1759791187,"id":"29689c1ef5366dfd01b5ecf9262b2da34cb02bafa1a8429bb6a490d04909d37d","kind":1,"pubkey":"832b77d5ecb09381ac37d75d6392424526d0923dced687b03d96ba03a5e3d55c","sig":"a5f9ea9b6867ee1be7ef756320fb156de99fe74187fbe6eb279481994e6633cea83e79d962ca01ff561be79172df352cc6a20e0db9cf7cc06c2e40b8361d374d","tags":[]}]
|
||||
[18:53:08.844] RECV nos.lol:443: ["EOSE","pool_1_1759791188"]
|
||||
[18:53:20.271] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"The Libertarians are the only political party promising to protect education choices in Australia. If you're a homeschooler the Libertarian party is the political party for you 🇦🇺 https://image.nostr.build/b5b36427856dfaee5ecdd4679977e6863a7cdc00ecef996d18b8b69b23c917c0.jpg ","created_at":1759791199,"id":"3da55d474418278f7ea7e5e08a853cc28d3dbf7e0f32f6f70636d6db80164b29","kind":1,"pubkey":"692b8a786c3be59c1e855a86e710215b35e32c55580500421407e740263f04cb","sig":"465d5819db53039d0c6b5d712c1bf1dc5611a4a3f49bfcf1ba6ac36d9f0eefd220f18a8cc43ef2522bf321882250ce5d660c42de4c8e839072ec5632269c2cbf","tags":[["imeta","url https://image.nostr.build/b5b36427856dfaee5ecdd4679977e6863a7cdc00ecef996d18b8b69b23c917c0.jpg","blurhash eXH-uqJU%zRP%f~BS4x]afxuR4xajEs:V@kW%LfkjZV[%MxttRNHkC","dim 1070x1338"],["r","https://image.nostr.build/b5b36427856dfaee5ecdd4679977e6863a7cdc00ecef996d18b8b69b23c917c0.jpg"]]}]
|
||||
[18:53:33.079] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"They are getting closer to realizing Bitcoin is the answer to physic.\n\nConservation without axiom is a big deal.","created_at":1759791212,"id":"87aadf3b737615ab0ba2208fdb7d61cbfa36f8bd4a31ed0c0c06f6818d724bf2","kind":1,"pubkey":"3c4f51561243524f307ed2ee272c7cf4a782404fbe3a176606043b6ad427ee77","sig":"8c832c83f784bdff2698021382b96acc63daa8ba605398044910cb4780222c4b37fb4eb90cb5322a2ae52dafd9db8d10fca3b7a99f3f2865386883427240c188","tags":[["e","81c9d02c125ae68ff39820d822de54ed6e8121e6804532f1bcd3afdd41e33c7a","wss://feeds.nostr.band/newstr","root"],["p","b4403b2415a020c20691bb18c51ada5acb64b71d2f60966cb3c78ba683542d4e","","mention"]]}]
|
||||
[18:53:34.774] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"一人称隣ニキで草 https://lumilumi.app/nevent1qvzqqqqqqypzq8tsnl8lcgukj2584c3t7mye5y835fc6ttz3s7kl9f3hvd2lf8d9qyt8wumn8ghj7un9d3shjtnwdaehgu3wvfskueqqypyhsn5gs4l2dnkwjgy2rz9kerad0lw773xngw640cq4stafvr57cp7sfq0","created_at":1759791214,"id":"178b376d3ed06cbd316f4daad7c5c215e2dc5d9586fe742b7e24cc4156949463","kind":1,"pubkey":"a19caaa8404721584746fb0e174cf971a94e0f51baaf4c4e8c6e54fa88985eaf","sig":"ec377113c06d030e0202511744496383ca4cfc5fb468d6106dc0391728511db175e31c8b18be17bacf19937ec4967e948ca31e5c793ae65197900fbc14b489a6","tags":[["r","https://lumilumi.app/nevent1qvzqqqqqqypzq8tsnl8lcgukj2584c3t7mye5y835fc6ttz3s7kl9f3hvd2lf8d9qyt8wumn8ghj7un9d3shjtnwdaehgu3wvfskueqqypyhsn5gs4l2dnkwjgy2rz9kerad0lw773xngw640cq4stafvr57cp7sfq0"],["client","lumilumi","31990:84b0c46ab699ac35eb2ca286470b85e081db2087cdef63932236c397417782f5:1727506446612","wss://relay.nostr.band"]]}]
|
||||
[18:53:35.041] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"How do people feel about #privacy apps that come from app stores vs direct from the developer? Even if the same developer? Does it make a difference? I'm trying to wrap my head around #Adnroid changes in the future...","created_at":1759791214,"id":"b997be2339010e78eb35747ad8b5e5e7ff9a352de0f6f143773e1a3888a865a9","kind":1,"pubkey":"8dc8688200b447ec2e4018ea5e42dc5d480940cb3f19ca8f361d28179dc4ba5e","sig":"67a82ec73da6eae1c157eaa8dea08b1ba0fccdc68c305c236150eeebd3f8aff1a897b52421b0821f9144dbbabe91032d7ad105d3b152f76638ae040f3e053668","tags":[]}]
|
||||
[18:53:44.914] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"please tell me this is satire. ","created_at":1759791224,"id":"93101e1379fc3564631bb22c1f92418e4a308453c00694229d66480e13e754f7","kind":1,"pubkey":"ff2ceb63833534d6e5b294b268501d62c53dcd2f05ee169f624ca1dc7cffe234","sig":"beee744a916041500fbe0cb2373f256948090ceafe3404a1b435544ea94ea392c70e5294cd56f02148556e4db6e6b81fc5274ac048200b8576610c04e4659278","tags":[["e","243c0fdc0879a37cb103dd8a9124889d40080741d67b04685409c7bfe2d445fb","wss://relay.mostr.pub/","root","11c63ffe2f02bce7e558470427deb0937d299af1140c4e24720804a1e0a7b08f"],["e","243c0fdc0879a37cb103dd8a9124889d40080741d67b04685409c7bfe2d445fb","wss://relay.mostr.pub/","reply","11c63ffe2f02bce7e558470427deb0937d299af1140c4e24720804a1e0a7b08f"],["p","11c63ffe2f02bce7e558470427deb0937d299af1140c4e24720804a1e0a7b08f"],["client","noStrudel","31990:266815e0c9210dfa324c6cba3573b14bee49da4209a9456f9484e5106cd408a5:1686066542546"]]}]
|
||||
[18:53:54.736] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"Transparency is in the favor of the people, not the greedy, slimy thieves who hide behind gov. https://image.nostr.build/8b5e9826ba98a2e90d0ae51a341bf20f6f525ea67ed4459d4dfe516c2a50c1ee.gif","created_at":1759791232,"id":"f0add365d13b283475f5aea939b6aa03b1dad5ecf363a7814249251f5e35c944","kind":1,"pubkey":"519d2fb3c354f44f5d4c7cdfc532633daf029bcc7d54beb0cb0f770646f8350f","sig":"6c7f8250e2f0efd421aa808760254a1d53d979c213efd23b0431c1ed634514ad6c7267762b2d3124df3e8925c0c557a899a4ddd4ec0f332276cde6b5ad6ea707","tags":[["r","https://image.nostr.build/8b5e9826ba98a2e90d0ae51a341bf20f6f525ea67ed4459d4dfe516c2a50c1ee.gif"],["imeta","url https://image.nostr.build/8b5e9826ba98a2e90d0ae51a341bf20f6f525ea67ed4459d4dfe516c2a50c1ee.gif","ox 8b5e9826ba98a2e90d0ae51a341bf20f6f525ea67ed4459d4dfe516c2a50c1ee","x 8707bfc0dacb4aafcc6f27d25639d93fe0c0ac4cafeea6ebbfb7c0aad9217ed9","m image/gif","dim 360x266","bh LH9s*cfk0eay~Vay4:ax-pfkIVWB","blurhash LH9s*cfk0eay~Vay4:ax-pfkIVWB","thumb https://image.nostr.build/thumb/8b5e9826ba98a2e90d0ae51a341bf20f6f525ea67ed4459d4dfe516c2a50c1ee.gif"]]}]
|
||||
[18:53:57.845] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"まずはピンク・フロイドを聴こうかなと思う","created_at":1759791237,"id":"fa7bfe34420bae341de5a1fc73bd2f4d5cb4283d6ab3b4c66dc697ebd5085ca5","kind":1,"pubkey":"f40832e26b1d12f8a27717b606996baef68bc4b6b86c4a35ca827f6fbcbf511e","sig":"0659b2963ff0e7c5e97e6a451ab572bcb0735f5371e8e20db3a82b0db555dcac449529c2cd0571f5c348ceb1f193a73a8437ec92062d4796b984c071e5b74542","tags":[]}]
|
||||
[18:54:03.482] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"#Bitcoin Block Art by Blockstr!\nHeight: 917953\nWeight: 3998184\nhttps://thebitcoinblockclock.com/blockstr/00000000000000000001707014fb634d4c3b44790f8e51da2fd89816d704d74c.png","created_at":1759791242,"id":"5342d9d61cf46692a331fe48c126bd671d8cf8b5fd0c99f8000669c7e606aeff","kind":1,"pubkey":"3fdf8b43d2e6eb59fc399f7cb1b81923d1dff0215d45a11e1c1f279827eaaad8","sig":"abaf2db95aafa23f9152cdf5f6a683c41696c732540d94e5da5ce0bece91dbc886c53a9d25be131eef93355456f857720eedbbfb26bb282eeba5347a360768a6","tags":[]}]
|
||||
[18:54:17.275] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"I would imagine hashrate goes up in northern countries where people mine4heat. But by how much I don’t know","created_at":1759791255,"id":"63e537f5b8047a577f3b3099295cc15916274e3e87997b65e54592e8166c1e30","kind":1,"pubkey":"18905d0a5d623ab81a98ba98c582bd5f57f2506c6b808905fc599d5a0b229b08","sig":"43c1092b80ab2fdd3c1dcdfd1a8bd47422be5b89d393ed8c8ad519d782a8ee9b137de2fe30ef6282c43e4882f9b57947d51bbaeb842fad03757d58c305d08af7","tags":[["p","18905d0a5d623ab81a98ba98c582bd5f57f2506c6b808905fc599d5a0b229b08","","mention"],["p","2b0ea532b3c3045d590c3dc982794d495d08bc497f554ccbcc30fcc8e9fbe2e9","","mention"],["e","d052b254eb2b402c4bc57035316885e4b5a39289083e69687408a5601123bea2","","root"],["e","6db869e46f7bdbaace29668104d99a0f1bbe91e2581a3b03dd53254f79159764","","reply"]]}]
|
||||
[18:54:18.389] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"I would recommend self-hosting with MusicSideProject.com to retain 100% control and 100% value sent to your songs. But if you're between Fountain and WL, Fountain shows the value splits in the RSS feed which is nice for listeners to see where their sats are flowing","created_at":1759791259,"id":"252cc6c9886902f4dfbdf9a120c14f7e1f4c7828fa3b0007445cbe93d1afe00b","kind":1,"pubkey":"2dfbe5cb955e4018b76e6542a3b8782144dd5f5735914db9921fd24c3b3d106d","sig":"f7db0d52d86e3099a0ddabe16eac696123f997ec6981a2fc51e32d24b3658eeb18e28086fd9460feccf96133b3a3b182ee31b6aaa420352c632607eb727f93fc","tags":[["alt","A short note: I would recommend self-hosting with MusicSideProje..."],["e","390de5cd45e4e0a8fae43ad405e6e194b4e23566ae66d0c145d7ea1e676cbb40","wss://nostr.oxtr.dev/","root","2b349bb9ae581ab6092365a16aef15c0ffc8f1a593e22f79b04fdb122543c1f9"],["e","49a3b9e700e684a0711184ecba7bb16b58ab5dba63feb0c27bb1ee6e80776331","wss://relay.damus.io/","reply","2c9a02818e5b9b4582c19823b69c55f5945e837aa9cc8342856f43b8ea61f979"],["p","2b349bb9ae581ab6092365a16aef15c0ffc8f1a593e22f79b04fdb122543c1f9","wss://zap.watch/"],["p","2c9a02818e5b9b4582c19823b69c55f5945e837aa9cc8342856f43b8ea61f979","wss://zap.watch/"],["r","https://musicsideproject.com/"]]}]
|
||||
[18:54:22.406] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"もう7時か、おはよう","created_at":1759791258,"id":"5fbce68be4f6fd34aca6115e294e5b790daf4ae210256556244b8ef33eff4b44","kind":1,"pubkey":"2bb2abbfc5892b7bda8f78d53682d913cc9a446b45e11929f0935d8fdfcb40bd","sig":"da49b572171e454aad4b9198443017dd25c1bd2e3313d5dac95931ea0db79198f47a29a31b0de7d9b20dd7e5fb0eac80916e00068b11a0c34c786e8ce50c8552","tags":[["e","cedbf52ea402b56b75069b64afeb7b9c9ce4e4063f607f76f715743e3a2b3707","","root","21ac29561b5de90cdc21995fc0707525cd78c8a52d87721ab681d3d609d1e2df"],["p","21ac29561b5de90cdc21995fc0707525cd78c8a52d87721ab681d3d609d1e2df"]]}]
|
||||
[18:54:23.521] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"Tell your family and friends you love them.\n\nI just had a friend pass away at 40 years old 😢\nHe was in pretty good health. I had known him for 35 years\n\nWe dont know when our time is up, enjoy the ride. Spend as much time as you can with loved ones. ","created_at":1759791262,"id":"dae2b9bb92b229a2b4e0cda21416a0545c5c87682c5e51bba909dce87db14298","kind":1,"pubkey":"7a6b8c7de171955c214ded7e35cc782cd6dddfd141abb1929c632f69348e6f49","sig":"bd3a18aa715ae940cc4744112abcc4e5b388bb6c351aaa559bd420e1b1336c1141824789645810d0f9ce7bcf49ef79922e6bee99367949ee14d616724fca928b","tags":[]}]
|
||||
[18:54:25.794] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"Personal experience agrees. Health took a very big, very noticeable dive after returning to America, from a country that bans the poisons.","created_at":1759791266,"id":"d31f8840cc9b9243e516aade74ced33dc37ce304f308ffaad19bcb786b6dd22c","kind":1,"pubkey":"55f573b651eff351db57b0601d23022d8c532f9825db10a5733ebf39be4aa21b","sig":"bdb42e44b1647cd77b59b8a3f01308aa8df89a1503b6c5d4984bd79ad4c20d50d6c4615589866985235719512519c54ec7ca863bca5f4f6cb8f9e19ae7e908cc","tags":[["e","0499a028291e2a5fff5cad342d19e985e4e1d16bc92170fd3ae4d700f5add047","","root"],["p","45b35521c312a5da4c2558703ad4be3d2e6d08c812551514c7a1eb7ab5fa0f04"],["p","da19b5d291e06ed09ba545bc5366408dc82ba8c9e9f97d32ea8ea60f1f6cbfaa"]]}]
|
||||
[18:54:26.842] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"08:02 @ 917,953","created_at":1759791266,"id":"17aaebb2d9e3d4b1cbb0eb085f0d031c91c8b982eb182b9c1737467ac312eba1","kind":1,"pubkey":"7c5f24e1c95f6f1f75555498f0019be1259a65c75ae851c235f7b15c9f88e0ee","sig":"5f216afb163dedf705741149fc4a7b1d3c7586eb61ff90aca6d78d08ef5e3a050e97c3e1628ad126b1d26b38c4236e3b6cd7b12799ef6129649d27f16bcd75b3","tags":[["t","bitcoin"],["t","moscowtime"],["t","nostr"]]}]
|
||||
[18:54:31.689] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":" \n\n\n\nhttps://blossom.primal.net/f69a2025125ef2f239621f96a9318f227df567f11db9aded4b57469e298e3317.mp4","created_at":1759791275,"id":"d89fe8ddd5874c17453cc6c8c9a23caae43d2e3fba6dc580d4b2527d3e00a240","kind":1,"pubkey":"e3a59924933c9f9f2df83449055eb3858f1480fdd0c5edb95df08bedcd2e6624","sig":"d1476ff7f3d3aac8aa042700fd4879f4d8c700294b0435fb63da314025aaaa03a5c2fc4e7ce2dd863a23779a007e6076b44b7286ca612f3d959a759b4ce79a03","tags":[]}]
|
||||
[18:54:36.744] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"去中心化“公链”的基本判断标准:\n节点门槛——低,众易上链;\n共识机制——牢,广受支持;\n代码开源——全,行为透明;\n治理机制——简,长链佐证;\n通证分布——散,没有寡头。\n#去中心化 #Decentralization\n#心空化\nhttps://cdn.nostrcheck.me/382d67ec6a31dc8d0c5b3099753a0044261a70e694aadd49eb58fce529cb3089/336be1277293f2d53fb97ab301957736be3e0a82eb16626e625bb9a2e1d26cd4.webp","created_at":1759791265,"id":"c37b409b25a77db2d43e19c85b957ceccd4650b04d1ad51130f00fc526a003ee","kind":1,"pubkey":"382d67ec6a31dc8d0c5b3099753a0044261a70e694aadd49eb58fce529cb3089","sig":"8116df2ad43a6126f68c6f26c5a8ba2a861227a76ff67123677113cf5f2319da7f558b94be946b80f7b49a9a13ed1edd422fa799346bf8cb6e9def3d9f4c9881","tags":[["imeta","url https://cdn.nostrcheck.me/382d67ec6a31dc8d0c5b3099753a0044261a70e694aadd49eb58fce529cb3089/336be1277293f2d53fb97ab301957736be3e0a82eb16626e625bb9a2e1d26cd4.webp","dim 827x814","sha256 336be1277293f2d53fb97ab301957736be3e0a82eb16626e625bb9a2e1d26cd4"],["t","去中心化"],["t","Decentralization"],["t","decentralization"],["t","心空化"],["client","Nostur","31990:9be0be0fc079548233231614e4e1efc9f28b0db398011efeecf05fe570e5dd33:1685868693432"]]}]
|
||||
[18:54:37.706] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"nostr:nprofile1qqszzf4msf9pxddg4hkre9a4ehdc67m9yhvapzg86chvs0enyzw80qqpypmhxue69uhky6t5vdhkjmndv9uxjmtpd35hxarn9ehkumrfdejj7qgkwaehxw309ajkgetw9ehx7um5wghxcctwvshsq3d5s2 \n\n","created_at":1759791276,"id":"4a5c21b093bc6b01a7535759e2bed8618876c00e97160d59867115306e8a7a7b","kind":1,"pubkey":"f6150173b5d6f079b43540d84a8a95d50cf01a48c9d6037984e3d9600d5522af","sig":"cfd5476a1a8477842fdbe66cc5d962f3a2e61f7909e94c36ce05557a4d740d0a69e99c72a4fedccfeb703083afdd3863b828688415efb5d25474dde74c61b1f1","tags":[["p","2126bb824a1335a8adec3c97b5cddb8d7b6525d9d08907d62ec83f33209c7780","wss://bitcoinmaximalists.online/","mention"],["p","683211bd155c7b764e4b99ba263a151d81209be7a566a2bb1971dc1bbd3b715e","wss://nos.lol/%20wss://nostr.land/%20%20avatar%20wss://nostr.wine/%20%20avatar%20wss://purplerelay.com/%20wss://relay.damus.io/%20wss://relay.snort.social/"],["e","be2ddd66a3d151a5fa6a38c0a43ffd4de5b20357abaf28b11954102348664624","wss://nos.lol/%20wss://nostr.land/%20%20avatar%20wss://nostr.wine/%20%20avatar%20wss://purplerelay.com/%20wss://relay.damus.io/%20wss://relay.snort.social/","root","683211bd155c7b764e4b99ba263a151d81209be7a566a2bb1971dc1bbd3b715e"]]}]
|
||||
[18:54:40.101] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"Day 67 of the $10/day DCA Tracker\n\nToday’s Results:\nBest Performer: Ethereum (+4.11%)\nWorst Performer: REIT (-0.94%)\n\n*graphic shows performance based on the total time since invested. It's tracking how a consistent $10/day DCA performs in each asset over time so you can see cumulative returns, not just daily fluctuations.\n\nFollow along for the ride!\nhttps://blossom.primal.net/659442920b5bf5f07fd154c896d20ea6433ad009e71cffe7aa83332837d98aec.png\n\n","created_at":1759791279,"id":"d9a4b37ca9b5c9fd6a7092cb95fd9d53f4b38ef257a9d4ca45a16ba94aaaf9ef","kind":1,"pubkey":"fac513a1ceded1eabc7407c12997485de8bbb28eddecefa016817fc8d4f407a5","sig":"e3ddab44f0f4e50ff45a72f56000a0d667290697017f93c093a901773c04c029c8060bd0261eff623c745b036332547d56ebcfb59498d9537f27d4fbdde569be","tags":[["r","wss://brb.io/"],["r","wss://eden.nostr.land/"],["r","wss://nos.lol/"],["r","wss://nostr.orangepill.dev/"],["r","wss://nostr.wine/"],["r","wss://relay.current.fyi/"],["r","wss://relay.damus.io/"],["r","wss://relay.snort.social/"],["client","Primal Studio"]]}]
|
||||
[18:54:42.363] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"GM ☕ 🌅👊","created_at":1759791281,"id":"0d89586ceaa523256647c1089090a7ad097b86a05c502f14772ea2a835fd9713","kind":1,"pubkey":"2822d3a6ae22a0af31193b03f2b2e84b5b2060bccdbefc0ee6696e128a3513bf","sig":"8205dfac792fd37147771dcd77a4b80fe68b6d41680727af7864090c4a5494270dbf7beccebe14e4fc773047ddc04a0b7af5bc3a72837a86085b86059b5a7f3c","tags":[["e","4024ce568c85efe6fc177367976f48bb080717589498fa7f8d5d4e1d0c17b3dd","wss://nostr-verified.wellorder.net","root"],["p","2c65940725bbf10bfbbf52b76c41606754441264f707d3d9cc1ceab86d73fd7f","","mention"]]}]
|
||||
[18:55:00.398] RECV nos.lol:443: ["EVENT","pool_1_1759791188",{"content":"✄------------ 7:55 ------------✄","created_at":1759791300,"id":"9ef33fbe34d3efe55cb9e4b4cd3b1281da6770c3b65ac761b260214937f3aa48","kind":1,"pubkey":"3ce2b51dca8b67b69c0ccb7c6a226437f7dbcc44a32426e70e52c78336fc72c7","sig":"dc83b7d49a2567540434f3041d5219e8c02c5f58566c6ff8a2b22e6c5e95391d1dc9f0244a149124d3aeeb2ec38515daef837656861cfd2f45f16dfc479fe30c","tags":[]}]
|
||||
[18:55:03.089] SEND localhost:7555: ["CLOSE", "pool_1_1759791188"]
|
||||
[18:55:03.089] SEND nos.lol:443: ["CLOSE", "pool_1_1759791188"]
|
||||
|
||||
=== NOSTR WebSocket Debug Log Started ===
|
||||
[05:52:03.588] SEND localhost:7555: ["REQ", "pool_1_1759830723", {
|
||||
"limit": 10
|
||||
}]
|
||||
[05:52:03.589] RECV localhost:7555: ["EVENT", "pool_1_1759830723", {
|
||||
"id": "8433206a6e00ff021ddd1f162af38bf724aabd6a7ca5fb9e57bda74f26d46e3a",
|
||||
"pubkey": "17323141f3a98b014f96ee85df59ee1ce6295e1502644c8e1f86439b20908198",
|
||||
"created_at": 1759687410,
|
||||
"kind": 1,
|
||||
"content": "Test post at 2025-10-05 14:03:30",
|
||||
"sig": "058a3bc6872e44e8d30cfba1d75267929ec3d73fbafa4c77f99897b700020a42478b51ee3bf4a4901beb376b9d3deca2854b3bca2f18eab169f9f5babdfeccb6",
|
||||
"tags": []
|
||||
}]
|
||||
[05:52:03.599] RECV localhost:7555: ["EVENT", "pool_1_1759830723", {
|
||||
"id": "ec98292f570049ed00ae541cba716dcf6c8252b523ce036d9d3d79946ff62927",
|
||||
"pubkey": "aa3b44608a9e52473a631d596c395dd297443237d56e34e31bdcd4272a1cf558",
|
||||
"created_at": 1759687283,
|
||||
"kind": 1,
|
||||
"content": "Test post at 2025-10-05 14:01:23",
|
||||
"sig": "3e5ffa59e881c4e772e3c0e65b26d5396b7687de2ae8e930b505fc2617ecfa4b392df32b3af41f8c989e5cb1eeb00d5c1fbedfba2729eb43a3838e49ebfffc8f",
|
||||
"tags": []
|
||||
}]
|
||||
[05:52:03.610] RECV localhost:7555: ["EVENT", "pool_1_1759830723", {
|
||||
"id": "c70d6c5c8745c088bd3b99e5f529bb69cd3d162cb9db28aa433a798fa8d811c3",
|
||||
"pubkey": "2a0c81450868960944c17b0f57f6aa2b5307b56bc8d9cc1b213674c561a3ab85",
|
||||
"created_at": 1759687249,
|
||||
"kind": 1,
|
||||
"content": "Test post at 2025-10-05 14:00:49",
|
||||
"sig": "76ac20012e84d49a528db140e48024d6a390033df75bfbad64e822208a6a4a3f863b84860da8bcf78c6d8c8933e30fba92f11ebca5f4602ba4428b10584c7ceb",
|
||||
"tags": []
|
||||
}]
|
||||
[05:52:03.620] RECV localhost:7555: ["EVENT", "pool_1_1759830723", {
|
||||
"id": "15dbe2cfe9237b93992a3b4de9d90224032e8b613031748da92c46c1aeaa2e9c",
|
||||
"pubkey": "7c2065299249ff1b4316c07af0d03d67ad145da80b907cc22aa608218c6d2687",
|
||||
"created_at": 1759687219,
|
||||
"kind": 1,
|
||||
"content": "Test post at 2025-10-05 14:00:19",
|
||||
"sig": "c0aa610658d8cfd76aca25894276a7f3f0573c2993429feff8b1ff10eeb5608ff18cfd79126de5f54e77f5b45a9d39f9006df65b56e4246e50cf72d212509569",
|
||||
"tags": []
|
||||
}]
|
||||
[05:52:03.631] RECV localhost:7555: ["EOSE", "pool_1_1759830723"]
|
||||
[05:52:31.497] SEND localhost:7555: ["REQ", "pool_2_1759830751", {
|
||||
"since": 1759830747,
|
||||
"limit": 10
|
||||
}]
|
||||
[05:52:31.498] RECV localhost:7555: ["EOSE", "pool_2_1759830751"]
|
||||
[14:47:46.012] SEND localhost:7555: ["REQ", "pool_1_1759687332", {
|
||||
"kinds": [1],
|
||||
"since": 1759687324,
|
||||
"limit": 10
|
||||
}]
|
||||
[14:47:48.014] SEND localhost:7555: ["REQ", "pool_1_1759830723", {
|
||||
"limit": 10
|
||||
}]
|
||||
[14:47:48.014] SEND localhost:7555: ["REQ", "pool_2_1759830751", {
|
||||
"since": 1759830747,
|
||||
"limit": 10
|
||||
}]
|
||||
|
||||
Binary file not shown.
@@ -8,6 +8,7 @@
|
||||
*/
|
||||
|
||||
#define _POSIX_C_SOURCE 200809L
|
||||
#define _DEFAULT_SOURCE
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
@@ -109,7 +110,7 @@ void* poll_thread_func(void* arg) {
|
||||
// Print menu
|
||||
void print_menu() {
|
||||
printf("\n=== NOSTR Relay Pool Test Menu ===\n");
|
||||
printf("1. Start Pool (wss://relay.laantungir.net)\n");
|
||||
printf("1. Start Pool (ws://localhost:7555)\n");
|
||||
printf("2. Stop Pool\n");
|
||||
printf("3. Add relay to pool\n");
|
||||
printf("4. Remove relay from pool\n");
|
||||
@@ -117,7 +118,8 @@ void print_menu() {
|
||||
printf("6. Remove subscription\n");
|
||||
printf("7. Show pool status\n");
|
||||
printf("8. Test reconnection (simulate disconnect)\n");
|
||||
printf("9. Exit\n");
|
||||
printf("9. Publish Event\n");
|
||||
printf("0. Exit\n");
|
||||
printf("Choice: ");
|
||||
}
|
||||
|
||||
@@ -405,11 +407,24 @@ void show_pool_status() {
|
||||
|
||||
printf("├── %s: %s\n", relay_urls[i], status_str);
|
||||
|
||||
// Show connection and publish error details
|
||||
const char* conn_error = nostr_relay_pool_get_relay_last_connection_error(pool, relay_urls[i]);
|
||||
const char* pub_error = nostr_relay_pool_get_relay_last_publish_error(pool, relay_urls[i]);
|
||||
|
||||
if (conn_error) {
|
||||
printf("│ ├── Connection error: %s\n", conn_error);
|
||||
}
|
||||
if (pub_error) {
|
||||
printf("│ ├── Last publish error: %s\n", pub_error);
|
||||
}
|
||||
|
||||
const nostr_relay_stats_t* stats = nostr_relay_pool_get_relay_stats(pool, relay_urls[i]);
|
||||
if (stats) {
|
||||
printf("│ ├── Events received: %d\n", stats->events_received);
|
||||
printf("│ ├── Connection attempts: %d\n", stats->connection_attempts);
|
||||
printf("│ ├── Connection failures: %d\n", stats->connection_failures);
|
||||
printf("│ ├── Events published: %d (OK: %d, Failed: %d)\n",
|
||||
stats->events_published, stats->events_published_ok, stats->events_published_failed);
|
||||
printf("│ ├── Ping latency: %.2f ms\n", stats->ping_latency_current);
|
||||
printf("│ └── Query latency: %.2f ms\n", stats->query_latency_avg);
|
||||
}
|
||||
@@ -422,6 +437,148 @@ void show_pool_status() {
|
||||
printf("\n");
|
||||
}
|
||||
|
||||
|
||||
// Async publish callback context
|
||||
typedef struct {
|
||||
int total_relays;
|
||||
int responses_received;
|
||||
int success_count;
|
||||
time_t start_time;
|
||||
} async_publish_context_t;
|
||||
|
||||
// Async publish callback - called for each relay response
|
||||
void async_publish_callback(const char* relay_url, const char* event_id,
|
||||
int success, const char* message, void* user_data) {
|
||||
async_publish_context_t* ctx = (async_publish_context_t*)user_data;
|
||||
|
||||
ctx->responses_received++;
|
||||
if (success) {
|
||||
ctx->success_count++;
|
||||
}
|
||||
|
||||
// Calculate elapsed time
|
||||
time_t now = time(NULL);
|
||||
double elapsed = difftime(now, ctx->start_time);
|
||||
|
||||
// Log to file with real-time feedback
|
||||
char timestamp[26];
|
||||
ctime_r(&now, timestamp);
|
||||
timestamp[24] = '\0';
|
||||
|
||||
if (success) {
|
||||
printf("✅ %s: Published successfully (%.1fs)\n", relay_url, elapsed);
|
||||
dprintf(log_fd, "[%s] ✅ ASYNC: %s published successfully (%.1fs)\n",
|
||||
timestamp, relay_url, elapsed);
|
||||
} else {
|
||||
printf("❌ %s: Failed - %s (%.1fs)\n", relay_url, message ? message : "unknown error", elapsed);
|
||||
dprintf(log_fd, "[%s] ❌ ASYNC: %s failed - %s (%.1fs)\n",
|
||||
timestamp, relay_url, message ? message : "unknown error", elapsed);
|
||||
}
|
||||
|
||||
// Show progress
|
||||
printf(" Progress: %d/%d responses received\n", ctx->responses_received, ctx->total_relays);
|
||||
|
||||
if (ctx->responses_received >= ctx->total_relays) {
|
||||
printf("\n🎉 All relays responded! Final result: %d/%d successful\n",
|
||||
ctx->success_count, ctx->total_relays);
|
||||
dprintf(log_fd, "[%s] 🎉 ASYNC: All relays responded - %d/%d successful\n\n",
|
||||
timestamp, ctx->success_count, ctx->total_relays);
|
||||
}
|
||||
}
|
||||
|
||||
// Publish test event with async callbacks
|
||||
void publish_event() {
|
||||
if (!pool) {
|
||||
printf("❌ Pool not started\n");
|
||||
return;
|
||||
}
|
||||
|
||||
printf("\n--- Publish Test Event ---\n");
|
||||
|
||||
// Generate random keypair
|
||||
unsigned char private_key[32], public_key[32];
|
||||
if (nostr_generate_keypair(private_key, public_key) != NOSTR_SUCCESS) {
|
||||
printf("❌ Failed to generate keypair\n");
|
||||
return;
|
||||
}
|
||||
|
||||
// Get current timestamp
|
||||
time_t now = time(NULL);
|
||||
|
||||
// Format content with date/time
|
||||
char content[256];
|
||||
struct tm* tm_info = localtime(&now);
|
||||
strftime(content, sizeof(content), "Test post at %Y-%m-%d %H:%M:%S", tm_info);
|
||||
|
||||
// Create kind 1 event
|
||||
cJSON* event = nostr_create_and_sign_event(1, content, NULL, private_key, now);
|
||||
if (!event) {
|
||||
printf("❌ Failed to create event\n");
|
||||
return;
|
||||
}
|
||||
|
||||
// Get relay URLs from pool
|
||||
char** relay_urls = NULL;
|
||||
nostr_pool_relay_status_t* statuses = NULL;
|
||||
int relay_count = nostr_relay_pool_list_relays(pool, &relay_urls, &statuses);
|
||||
|
||||
if (relay_count <= 0) {
|
||||
printf("❌ No relays in pool\n");
|
||||
cJSON_Delete(event);
|
||||
return;
|
||||
}
|
||||
|
||||
printf("📤 Publishing event to %d relay(s)...\n", relay_count);
|
||||
printf("Watch for real-time responses below:\n\n");
|
||||
|
||||
// Setup callback context
|
||||
async_publish_context_t ctx = {0};
|
||||
ctx.total_relays = relay_count;
|
||||
ctx.start_time = time(NULL);
|
||||
|
||||
// Log the event
|
||||
char* event_json = cJSON_Print(event);
|
||||
char timestamp[26];
|
||||
ctime_r(&now, timestamp);
|
||||
timestamp[24] = '\0';
|
||||
dprintf(log_fd, "[%s] 📤 Publishing test event\n", timestamp);
|
||||
dprintf(log_fd, "Event: %s\n\n", event_json);
|
||||
free(event_json);
|
||||
|
||||
// Publish using async function
|
||||
int sent_count = nostr_relay_pool_publish_async(pool, (const char**)relay_urls,
|
||||
relay_count, event,
|
||||
async_publish_callback, &ctx);
|
||||
|
||||
if (sent_count > 0) {
|
||||
printf("📡 Event sent to %d/%d relays, waiting for responses...\n\n",
|
||||
sent_count, relay_count);
|
||||
|
||||
// Wait for all responses or timeout (10 seconds)
|
||||
time_t wait_start = time(NULL);
|
||||
while (ctx.responses_received < ctx.total_relays &&
|
||||
(time(NULL) - wait_start) < 10) {
|
||||
// Let the polling thread process messages
|
||||
usleep(100000); // 100ms
|
||||
}
|
||||
|
||||
if (ctx.responses_received < ctx.total_relays) {
|
||||
printf("\n⏰ Timeout reached - %d/%d relays responded\n",
|
||||
ctx.responses_received, ctx.total_relays);
|
||||
}
|
||||
} else {
|
||||
printf("❌ Failed to send event to any relays\n");
|
||||
}
|
||||
|
||||
// Cleanup
|
||||
for (int i = 0; i < relay_count; i++) {
|
||||
free(relay_urls[i]);
|
||||
}
|
||||
free(relay_urls);
|
||||
free(statuses);
|
||||
cJSON_Delete(event);
|
||||
}
|
||||
|
||||
int main() {
|
||||
// Setup logging to file
|
||||
log_fd = open("pool.log", O_WRONLY | O_CREAT | O_TRUNC, 0644);
|
||||
@@ -489,14 +646,14 @@ int main() {
|
||||
break;
|
||||
}
|
||||
|
||||
if (nostr_relay_pool_add_relay(pool, "wss://relay.laantungir.net") != NOSTR_SUCCESS) {
|
||||
if (nostr_relay_pool_add_relay(pool, "ws://localhost:7555") != NOSTR_SUCCESS) {
|
||||
printf("❌ Failed to add default relay\n");
|
||||
nostr_relay_pool_destroy(pool);
|
||||
pool = NULL;
|
||||
break;
|
||||
}
|
||||
|
||||
printf("✅ Pool started with wss://relay.laantungir.net\n");
|
||||
printf("✅ Pool started with ws://localhost:7555\n");
|
||||
|
||||
now = time(NULL);
|
||||
ctime_r(&now, timestamp);
|
||||
@@ -544,13 +701,49 @@ int main() {
|
||||
if (url && strlen(url) > 0) {
|
||||
if (nostr_relay_pool_add_relay(pool, url) == NOSTR_SUCCESS) {
|
||||
printf("✅ Relay added: %s\n", url);
|
||||
printf("⏳ Attempting to connect...\n");
|
||||
|
||||
// Give it a moment to attempt connection
|
||||
sleep(2);
|
||||
|
||||
// Check connection status and show any errors
|
||||
nostr_pool_relay_status_t status = nostr_relay_pool_get_relay_status(pool, url);
|
||||
const char* error_msg = nostr_relay_pool_get_relay_last_connection_error(pool, url);
|
||||
|
||||
switch (status) {
|
||||
case NOSTR_POOL_RELAY_CONNECTED:
|
||||
printf("🟢 Successfully connected to %s\n", url);
|
||||
break;
|
||||
case NOSTR_POOL_RELAY_CONNECTING:
|
||||
printf("🟡 Still connecting to %s...\n", url);
|
||||
break;
|
||||
case NOSTR_POOL_RELAY_DISCONNECTED:
|
||||
printf("⚪ Disconnected from %s\n", url);
|
||||
if (error_msg) {
|
||||
printf(" Last error: %s\n", error_msg);
|
||||
}
|
||||
break;
|
||||
case NOSTR_POOL_RELAY_ERROR:
|
||||
printf("🔴 Connection error for %s\n", url);
|
||||
if (error_msg) {
|
||||
printf(" Error details: %s\n", error_msg);
|
||||
}
|
||||
break;
|
||||
default:
|
||||
printf("❓ Unknown status for %s\n", url);
|
||||
break;
|
||||
}
|
||||
|
||||
now = time(NULL);
|
||||
ctime_r(&now, timestamp);
|
||||
timestamp[24] = '\0';
|
||||
dprintf(log_fd, "[%s] ➕ Relay added: %s\n\n", timestamp, url);
|
||||
dprintf(log_fd, "[%s] ➕ Relay added: %s (status: %d)\n", timestamp, url, status);
|
||||
if (error_msg) {
|
||||
dprintf(log_fd, " Connection error: %s\n", error_msg);
|
||||
}
|
||||
dprintf(log_fd, "\n");
|
||||
} else {
|
||||
printf("❌ Failed to add relay\n");
|
||||
printf("❌ Failed to add relay to pool\n");
|
||||
}
|
||||
}
|
||||
free(url);
|
||||
@@ -655,7 +848,11 @@ int main() {
|
||||
break;
|
||||
}
|
||||
|
||||
case '9': // Exit
|
||||
case '9': // Publish Event
|
||||
publish_event();
|
||||
break;
|
||||
|
||||
case '0': // Exit
|
||||
running = 0;
|
||||
break;
|
||||
|
||||
|
||||
@@ -41,6 +41,7 @@
|
||||
#define NOSTR_POOL_SUBSCRIPTION_ID_SIZE 32
|
||||
#define NOSTR_POOL_PING_INTERVAL 59 // 59 seconds - keeps connections alive
|
||||
#define NOSTR_POOL_MAX_PENDING_SUBSCRIPTIONS 8 // Max concurrent subscription timings per relay
|
||||
#define NOSTR_POOL_MAX_PENDING_PUBLISHES 32 // Max concurrent publish operations
|
||||
|
||||
// High-resolution timing helper
|
||||
static double get_current_time_ms(void) {
|
||||
@@ -60,6 +61,17 @@ typedef struct subscription_timing {
|
||||
int active;
|
||||
} subscription_timing_t;
|
||||
|
||||
// Publish operation tracking for async callbacks
|
||||
typedef struct publish_operation {
|
||||
char event_id[65]; // Event ID being published
|
||||
publish_response_callback_t callback; // User callback function
|
||||
void* user_data; // User data for callback
|
||||
time_t publish_time; // When publish was initiated
|
||||
int pending_relay_count; // Number of relays still pending response
|
||||
char** pending_relay_urls; // URLs of relays still pending
|
||||
int total_relay_count; // Total number of relays for this publish
|
||||
} publish_operation_t;
|
||||
|
||||
// Internal structures
|
||||
typedef struct relay_connection {
|
||||
char* url;
|
||||
@@ -147,6 +159,10 @@ struct nostr_relay_pool {
|
||||
nostr_pool_subscription_t* subscriptions[NOSTR_POOL_MAX_SUBSCRIPTIONS];
|
||||
int subscription_count;
|
||||
|
||||
// Active publish operations for async callbacks
|
||||
publish_operation_t* publish_operations[NOSTR_POOL_MAX_PENDING_PUBLISHES];
|
||||
int publish_operation_count;
|
||||
|
||||
// Pool-wide settings
|
||||
int default_timeout_ms;
|
||||
};
|
||||
@@ -217,6 +233,122 @@ static double remove_subscription_timing(relay_connection_t* relay, const char*
|
||||
return -1.0; // Not found
|
||||
}
|
||||
|
||||
// Helper functions for managing publish operations
|
||||
static int add_publish_operation(nostr_relay_pool_t* pool, const char* event_id,
|
||||
const char** relay_urls, int relay_count,
|
||||
publish_response_callback_t callback, void* user_data) {
|
||||
if (!pool || !event_id || !relay_urls || relay_count <= 0) return -1;
|
||||
|
||||
// Check if we have space for another operation
|
||||
if (pool->publish_operation_count >= NOSTR_POOL_MAX_PENDING_PUBLISHES) {
|
||||
return -1; // No space available
|
||||
}
|
||||
|
||||
// Create new publish operation
|
||||
publish_operation_t* op = calloc(1, sizeof(publish_operation_t));
|
||||
if (!op) return -1;
|
||||
|
||||
// Copy event ID
|
||||
strncpy(op->event_id, event_id, sizeof(op->event_id) - 1);
|
||||
op->event_id[sizeof(op->event_id) - 1] = '\0';
|
||||
|
||||
// Set callback and user data
|
||||
op->callback = callback;
|
||||
op->user_data = user_data;
|
||||
op->publish_time = time(NULL);
|
||||
op->total_relay_count = relay_count;
|
||||
op->pending_relay_count = relay_count;
|
||||
|
||||
// Copy relay URLs
|
||||
op->pending_relay_urls = malloc(relay_count * sizeof(char*));
|
||||
if (!op->pending_relay_urls) {
|
||||
free(op);
|
||||
return -1;
|
||||
}
|
||||
|
||||
for (int i = 0; i < relay_count; i++) {
|
||||
op->pending_relay_urls[i] = strdup(relay_urls[i]);
|
||||
if (!op->pending_relay_urls[i]) {
|
||||
// Cleanup on failure
|
||||
for (int j = 0; j < i; j++) {
|
||||
free(op->pending_relay_urls[j]);
|
||||
}
|
||||
free(op->pending_relay_urls);
|
||||
free(op);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
// Add to pool
|
||||
pool->publish_operations[pool->publish_operation_count++] = op;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static publish_operation_t* find_publish_operation(nostr_relay_pool_t* pool, const char* event_id) {
|
||||
if (!pool || !event_id) return NULL;
|
||||
|
||||
for (int i = 0; i < pool->publish_operation_count; i++) {
|
||||
if (pool->publish_operations[i] &&
|
||||
strcmp(pool->publish_operations[i]->event_id, event_id) == 0) {
|
||||
return pool->publish_operations[i];
|
||||
}
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static void remove_publish_operation(nostr_relay_pool_t* pool, const char* event_id) {
|
||||
if (!pool || !event_id) return;
|
||||
|
||||
for (int i = 0; i < pool->publish_operation_count; i++) {
|
||||
if (pool->publish_operations[i] &&
|
||||
strcmp(pool->publish_operations[i]->event_id, event_id) == 0) {
|
||||
|
||||
publish_operation_t* op = pool->publish_operations[i];
|
||||
|
||||
// Free relay URLs (only non-NULL ones)
|
||||
if (op->pending_relay_urls) {
|
||||
for (int j = 0; j < op->total_relay_count; j++) {
|
||||
if (op->pending_relay_urls[j]) {
|
||||
free(op->pending_relay_urls[j]);
|
||||
op->pending_relay_urls[j] = NULL;
|
||||
}
|
||||
}
|
||||
free(op->pending_relay_urls);
|
||||
op->pending_relay_urls = NULL;
|
||||
}
|
||||
free(op);
|
||||
|
||||
// Shift remaining operations
|
||||
for (int j = i; j < pool->publish_operation_count - 1; j++) {
|
||||
pool->publish_operations[j] = pool->publish_operations[j + 1];
|
||||
}
|
||||
pool->publish_operations[--pool->publish_operation_count] = NULL;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int remove_relay_from_publish_operation(publish_operation_t* op, const char* relay_url) {
|
||||
if (!op || !relay_url) return -1;
|
||||
|
||||
for (int i = 0; i < op->pending_relay_count; i++) {
|
||||
if (op->pending_relay_urls[i] && strcmp(op->pending_relay_urls[i], relay_url) == 0) {
|
||||
// Free this relay URL
|
||||
free(op->pending_relay_urls[i]);
|
||||
op->pending_relay_urls[i] = NULL; // Mark as freed
|
||||
|
||||
// Shift remaining URLs
|
||||
for (int j = i; j < op->pending_relay_count - 1; j++) {
|
||||
op->pending_relay_urls[j] = op->pending_relay_urls[j + 1];
|
||||
}
|
||||
op->pending_relay_urls[op->pending_relay_count - 1] = NULL; // Clear the last slot
|
||||
op->pending_relay_count--;
|
||||
return op->pending_relay_count; // Return remaining count
|
||||
}
|
||||
}
|
||||
return -1; // Relay not found
|
||||
}
|
||||
|
||||
// Helper function to ensure relay connection
|
||||
static int ensure_relay_connection(relay_connection_t* relay) {
|
||||
if (!relay) {
|
||||
@@ -559,6 +691,20 @@ void nostr_relay_pool_destroy(nostr_relay_pool_t* pool) {
|
||||
}
|
||||
}
|
||||
|
||||
// Clean up all pending publish operations
|
||||
for (int i = 0; i < pool->publish_operation_count; i++) {
|
||||
if (pool->publish_operations[i]) {
|
||||
publish_operation_t* op = pool->publish_operations[i];
|
||||
|
||||
// Free relay URLs
|
||||
for (int j = 0; j < op->total_relay_count; j++) {
|
||||
free(op->pending_relay_urls[j]);
|
||||
}
|
||||
free(op->pending_relay_urls);
|
||||
free(op);
|
||||
}
|
||||
}
|
||||
|
||||
// Close all relay connections
|
||||
for (int i = 0; i < pool->relay_count; i++) {
|
||||
if (pool->relays[i]) {
|
||||
@@ -902,25 +1048,48 @@ static void process_relay_message(nostr_relay_pool_t* pool, relay_connection_t*
|
||||
} else if (strcmp(msg_type, "OK") == 0) {
|
||||
// Handle OK response: ["OK", event_id, true/false, message]
|
||||
if (cJSON_IsArray(parsed) && cJSON_GetArraySize(parsed) >= 3) {
|
||||
cJSON* event_id_json = cJSON_GetArrayItem(parsed, 1);
|
||||
cJSON* success_flag = cJSON_GetArrayItem(parsed, 2);
|
||||
|
||||
if (cJSON_IsBool(success_flag)) {
|
||||
if (cJSON_IsTrue(success_flag)) {
|
||||
if (cJSON_IsString(event_id_json) && cJSON_IsBool(success_flag)) {
|
||||
const char* event_id = cJSON_GetStringValue(event_id_json);
|
||||
int success = cJSON_IsTrue(success_flag);
|
||||
const char* error_message = NULL;
|
||||
|
||||
// Extract error message if available
|
||||
if (!success && cJSON_GetArraySize(parsed) >= 4) {
|
||||
cJSON* error_msg = cJSON_GetArrayItem(parsed, 3);
|
||||
if (cJSON_IsString(error_msg)) {
|
||||
error_message = cJSON_GetStringValue(error_msg);
|
||||
}
|
||||
}
|
||||
|
||||
// Update relay statistics
|
||||
if (success) {
|
||||
relay->stats.events_published_ok++;
|
||||
} else {
|
||||
relay->stats.events_published_failed++;
|
||||
// Store error message if available
|
||||
if (cJSON_GetArraySize(parsed) >= 4) {
|
||||
cJSON* error_msg = cJSON_GetArrayItem(parsed, 3);
|
||||
if (cJSON_IsString(error_msg)) {
|
||||
const char* msg = cJSON_GetStringValue(error_msg);
|
||||
if (msg) {
|
||||
strncpy(relay->last_publish_error, msg,
|
||||
sizeof(relay->last_publish_error) - 1);
|
||||
relay->last_publish_error[sizeof(relay->last_publish_error) - 1] = '\0';
|
||||
relay->last_publish_error_time = time(NULL);
|
||||
}
|
||||
}
|
||||
// Store error message for legacy API
|
||||
if (error_message) {
|
||||
strncpy(relay->last_publish_error, error_message,
|
||||
sizeof(relay->last_publish_error) - 1);
|
||||
relay->last_publish_error[sizeof(relay->last_publish_error) - 1] = '\0';
|
||||
relay->last_publish_error_time = time(NULL);
|
||||
}
|
||||
}
|
||||
|
||||
// Check for async publish operation callback
|
||||
publish_operation_t* op = find_publish_operation(pool, event_id);
|
||||
if (op && op->callback) {
|
||||
// Call the user's callback
|
||||
op->callback(relay->url, event_id, success, error_message, op->user_data);
|
||||
|
||||
// Remove this relay from the pending list
|
||||
int remaining = remove_relay_from_publish_operation(op, relay->url);
|
||||
|
||||
// If no more relays pending, remove the operation
|
||||
if (remaining == 0) {
|
||||
remove_publish_operation(pool, event_id);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1113,16 +1282,35 @@ cJSON* nostr_relay_pool_get_event(
|
||||
return result;
|
||||
}
|
||||
|
||||
int nostr_relay_pool_publish(
|
||||
|
||||
int nostr_relay_pool_publish_async(
|
||||
nostr_relay_pool_t* pool,
|
||||
const char** relay_urls,
|
||||
int relay_count,
|
||||
cJSON* event) {
|
||||
cJSON* event,
|
||||
publish_response_callback_t callback,
|
||||
void* user_data) {
|
||||
|
||||
if (!pool || !relay_urls || relay_count <= 0 || !event) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
// Extract event ID for tracking
|
||||
cJSON* event_id_json = cJSON_GetObjectItem(event, "id");
|
||||
if (!event_id_json || !cJSON_IsString(event_id_json)) {
|
||||
return -1; // Event must have an ID
|
||||
}
|
||||
const char* event_id = cJSON_GetStringValue(event_id_json);
|
||||
|
||||
// Add publish operation for tracking (only if callback provided)
|
||||
publish_operation_t* op = NULL;
|
||||
if (callback) {
|
||||
if (add_publish_operation(pool, event_id, relay_urls, relay_count, callback, user_data) != 0) {
|
||||
return -1; // Failed to add operation
|
||||
}
|
||||
op = find_publish_operation(pool, event_id);
|
||||
}
|
||||
|
||||
int success_count = 0;
|
||||
|
||||
for (int i = 0; i < relay_count; i++) {
|
||||
@@ -1135,55 +1323,35 @@ int nostr_relay_pool_publish(
|
||||
}
|
||||
|
||||
if (relay && ensure_relay_connection(relay) == 0) {
|
||||
double start_time_ms = get_current_time_ms();
|
||||
|
||||
// Send EVENT message
|
||||
if (nostr_relay_send_event(relay->ws_client, event) >= 0) {
|
||||
relay->stats.events_published++;
|
||||
|
||||
// Wait for OK response
|
||||
char buffer[1024];
|
||||
time_t wait_start = time(NULL);
|
||||
int got_response = 0;
|
||||
|
||||
while (time(NULL) - wait_start < 5 && !got_response) { // 5 second timeout
|
||||
int len = nostr_ws_receive(relay->ws_client, buffer, sizeof(buffer) - 1, 1000);
|
||||
if (len > 0) {
|
||||
buffer[len] = '\0';
|
||||
|
||||
char* msg_type = NULL;
|
||||
cJSON* parsed = NULL;
|
||||
if (nostr_parse_relay_message(buffer, &msg_type, &parsed) == 0) {
|
||||
if (msg_type && strcmp(msg_type, "OK") == 0) {
|
||||
// Handle OK response
|
||||
if (cJSON_IsArray(parsed) && cJSON_GetArraySize(parsed) >= 3) {
|
||||
cJSON* success_flag = cJSON_GetArrayItem(parsed, 2);
|
||||
if (cJSON_IsBool(success_flag) && cJSON_IsTrue(success_flag)) {
|
||||
success_count++;
|
||||
relay->stats.events_published_ok++;
|
||||
|
||||
// Update publish latency statistics
|
||||
double latency_ms = get_current_time_ms() - start_time_ms;
|
||||
if (relay->stats.publish_samples == 0) {
|
||||
relay->stats.publish_latency_avg = latency_ms;
|
||||
} else {
|
||||
relay->stats.publish_latency_avg =
|
||||
(relay->stats.publish_latency_avg * relay->stats.publish_samples + latency_ms) /
|
||||
(relay->stats.publish_samples + 1);
|
||||
}
|
||||
relay->stats.publish_samples++;
|
||||
} else {
|
||||
relay->stats.events_published_failed++;
|
||||
}
|
||||
}
|
||||
got_response = 1;
|
||||
}
|
||||
if (msg_type) free(msg_type);
|
||||
if (parsed) cJSON_Delete(parsed);
|
||||
}
|
||||
success_count++;
|
||||
} else {
|
||||
// If send failed and we have a callback, notify immediately
|
||||
if (callback && op) {
|
||||
callback(relay_urls[i], event_id, 0, "Failed to send event to relay", user_data);
|
||||
|
||||
// Remove this relay from the pending operation
|
||||
int remaining = remove_relay_from_publish_operation(op, relay_urls[i]);
|
||||
if (remaining == 0) {
|
||||
remove_publish_operation(pool, event_id);
|
||||
op = NULL; // Mark as removed to prevent double-free
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
// Connection failed - notify callback immediately if provided
|
||||
if (callback && op) {
|
||||
callback(relay_urls[i], event_id, 0, "Failed to connect to relay", user_data);
|
||||
|
||||
// Remove this relay from the pending operation
|
||||
int remaining = remove_relay_from_publish_operation(op, relay_urls[i]);
|
||||
if (remaining == 0) {
|
||||
remove_publish_operation(pool, event_id);
|
||||
op = NULL; // Mark as removed to prevent double-free
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -13,7 +13,7 @@ extern "C" {
|
||||
#endif
|
||||
|
||||
// NIP-04 constants
|
||||
// #define NOSTR_NIP04_MAX_PLAINTEXT_SIZE 65535
|
||||
// #define NOSTR_NIP04_MAX_PLAINTEXT_SIZE 1048576 // 1MB
|
||||
// NIP-04 Constants
|
||||
// #define NOSTR_NIP04_MAX_PLAINTEXT_SIZE 16777216 // 16MB
|
||||
// #define NOSTR_NIP04_MAX_ENCRYPTED_SIZE 22369621 // ~21.3MB (accounts for base64 overhead + IV)
|
||||
|
||||
@@ -258,11 +258,12 @@ cJSON* nostr_nip17_create_relay_list_event(const char** relay_urls,
|
||||
* NIP-17: Send a direct message to recipients
|
||||
*/
|
||||
int nostr_nip17_send_dm(cJSON* dm_event,
|
||||
const char** recipient_pubkeys,
|
||||
int num_recipients,
|
||||
const unsigned char* sender_private_key,
|
||||
cJSON** gift_wraps_out,
|
||||
int max_gift_wraps) {
|
||||
const char** recipient_pubkeys,
|
||||
int num_recipients,
|
||||
const unsigned char* sender_private_key,
|
||||
cJSON** gift_wraps_out,
|
||||
int max_gift_wraps,
|
||||
long max_delay_sec) {
|
||||
if (!dm_event || !recipient_pubkeys || num_recipients <= 0 ||
|
||||
!sender_private_key || !gift_wraps_out || max_gift_wraps <= 0) {
|
||||
return -1;
|
||||
@@ -278,13 +279,13 @@ int nostr_nip17_send_dm(cJSON* dm_event,
|
||||
}
|
||||
|
||||
// Create seal for this recipient
|
||||
cJSON* seal = nostr_nip59_create_seal(dm_event, sender_private_key, recipient_public_key);
|
||||
cJSON* seal = nostr_nip59_create_seal(dm_event, sender_private_key, recipient_public_key, max_delay_sec);
|
||||
if (!seal) {
|
||||
continue; // Skip if sealing fails
|
||||
}
|
||||
|
||||
// Create gift wrap for this recipient
|
||||
cJSON* gift_wrap = nostr_nip59_create_gift_wrap(seal, recipient_pubkeys[i]);
|
||||
cJSON* gift_wrap = nostr_nip59_create_gift_wrap(seal, recipient_pubkeys[i], max_delay_sec);
|
||||
cJSON_Delete(seal); // Seal is now wrapped
|
||||
|
||||
if (!gift_wrap) {
|
||||
@@ -303,10 +304,10 @@ int nostr_nip17_send_dm(cJSON* dm_event,
|
||||
nostr_bytes_to_hex(sender_public_key, 32, sender_pubkey_hex);
|
||||
|
||||
// Create seal for sender
|
||||
cJSON* sender_seal = nostr_nip59_create_seal(dm_event, sender_private_key, sender_public_key);
|
||||
cJSON* sender_seal = nostr_nip59_create_seal(dm_event, sender_private_key, sender_public_key, max_delay_sec);
|
||||
if (sender_seal) {
|
||||
// Create gift wrap for sender
|
||||
cJSON* sender_gift_wrap = nostr_nip59_create_gift_wrap(sender_seal, sender_pubkey_hex);
|
||||
cJSON* sender_gift_wrap = nostr_nip59_create_gift_wrap(sender_seal, sender_pubkey_hex, max_delay_sec);
|
||||
cJSON_Delete(sender_seal);
|
||||
|
||||
if (sender_gift_wrap) {
|
||||
|
||||
@@ -97,6 +97,7 @@ cJSON* nostr_nip17_create_relay_list_event(const char** relay_urls,
|
||||
* @param sender_private_key 32-byte sender private key
|
||||
* @param gift_wraps_out Array to store resulting gift wrap events (caller must free)
|
||||
* @param max_gift_wraps Maximum number of gift wraps to create
|
||||
* @param max_delay_sec Maximum random timestamp delay in seconds (0 = no randomization)
|
||||
* @return Number of gift wrap events created, or -1 on error
|
||||
*/
|
||||
int nostr_nip17_send_dm(cJSON* dm_event,
|
||||
@@ -104,7 +105,8 @@ int nostr_nip17_send_dm(cJSON* dm_event,
|
||||
int num_recipients,
|
||||
const unsigned char* sender_private_key,
|
||||
cJSON** gift_wraps_out,
|
||||
int max_gift_wraps);
|
||||
int max_gift_wraps,
|
||||
long max_delay_sec);
|
||||
|
||||
/**
|
||||
* NIP-17: Receive and decrypt a direct message
|
||||
|
||||
@@ -13,7 +13,7 @@ extern "C" {
|
||||
#endif
|
||||
|
||||
// NIP-44 constants
|
||||
// #define NOSTR_NIP44_MAX_PLAINTEXT_SIZE 65535
|
||||
// #define NOSTR_NIP44_MAX_PLAINTEXT_SIZE 1048576
|
||||
|
||||
/**
|
||||
* NIP-44: Encrypt a message using ECDH + ChaCha20 + HMAC
|
||||
|
||||
@@ -26,12 +26,18 @@ static void memory_clear(const void *p, size_t len) {
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a random timestamp within 2 days in the past (as per NIP-59 spec)
|
||||
* Create a random timestamp within max_delay_sec in the past (configurable)
|
||||
*/
|
||||
static time_t random_past_timestamp(void) {
|
||||
static time_t random_past_timestamp(long max_delay_sec) {
|
||||
time_t now = time(NULL);
|
||||
// Random time up to 2 days (172800 seconds) in the past
|
||||
long random_offset = (long)(rand() % 172800);
|
||||
|
||||
// If max_delay_sec is 0, return current timestamp (no randomization)
|
||||
if (max_delay_sec == 0) {
|
||||
return now;
|
||||
}
|
||||
|
||||
// Random time up to max_delay_sec in the past
|
||||
long random_offset = (long)(rand() % max_delay_sec);
|
||||
return now - random_offset;
|
||||
}
|
||||
|
||||
@@ -104,8 +110,8 @@ cJSON* nostr_nip59_create_rumor(int kind, const char* content, cJSON* tags,
|
||||
return NULL;
|
||||
}
|
||||
|
||||
// Use provided timestamp or random past timestamp
|
||||
time_t event_time = (created_at == 0) ? random_past_timestamp() : created_at;
|
||||
// Use provided timestamp or random past timestamp (default to 0 for compatibility)
|
||||
time_t event_time = (created_at == 0) ? random_past_timestamp(0) : created_at;
|
||||
|
||||
// Create event structure (without id and sig - that's what makes it a rumor)
|
||||
cJSON* rumor = cJSON_CreateObject();
|
||||
@@ -142,7 +148,7 @@ cJSON* nostr_nip59_create_rumor(int kind, const char* content, cJSON* tags,
|
||||
* NIP-59: Create a seal (kind 13) wrapping a rumor
|
||||
*/
|
||||
cJSON* nostr_nip59_create_seal(cJSON* rumor, const unsigned char* sender_private_key,
|
||||
const unsigned char* recipient_public_key) {
|
||||
const unsigned char* recipient_public_key, long max_delay_sec) {
|
||||
if (!rumor || !sender_private_key || !recipient_public_key) {
|
||||
return NULL;
|
||||
}
|
||||
@@ -178,7 +184,7 @@ cJSON* nostr_nip59_create_seal(cJSON* rumor, const unsigned char* sender_private
|
||||
return NULL;
|
||||
}
|
||||
|
||||
time_t seal_time = random_past_timestamp();
|
||||
time_t seal_time = random_past_timestamp(max_delay_sec);
|
||||
|
||||
cJSON_AddStringToObject(seal, "pubkey", sender_pubkey_hex);
|
||||
cJSON_AddNumberToObject(seal, "created_at", (double)seal_time);
|
||||
@@ -217,7 +223,7 @@ cJSON* nostr_nip59_create_seal(cJSON* rumor, const unsigned char* sender_private
|
||||
/**
|
||||
* NIP-59: Create a gift wrap (kind 1059) wrapping a seal
|
||||
*/
|
||||
cJSON* nostr_nip59_create_gift_wrap(cJSON* seal, const char* recipient_public_key_hex) {
|
||||
cJSON* nostr_nip59_create_gift_wrap(cJSON* seal, const char* recipient_public_key_hex, long max_delay_sec) {
|
||||
if (!seal || !recipient_public_key_hex) {
|
||||
return NULL;
|
||||
}
|
||||
@@ -272,7 +278,7 @@ cJSON* nostr_nip59_create_gift_wrap(cJSON* seal, const char* recipient_public_ke
|
||||
return NULL;
|
||||
}
|
||||
|
||||
time_t wrap_time = random_past_timestamp();
|
||||
time_t wrap_time = random_past_timestamp(max_delay_sec);
|
||||
|
||||
cJSON_AddStringToObject(gift_wrap, "pubkey", random_pubkey_hex);
|
||||
cJSON_AddNumberToObject(gift_wrap, "created_at", (double)wrap_time);
|
||||
|
||||
@@ -33,19 +33,21 @@ cJSON* nostr_nip59_create_rumor(int kind, const char* content, cJSON* tags,
|
||||
* @param rumor The rumor event to seal (cJSON object)
|
||||
* @param sender_private_key 32-byte sender private key
|
||||
* @param recipient_public_key 32-byte recipient public key (x-only)
|
||||
* @param max_delay_sec Maximum random timestamp delay in seconds (0 = no randomization)
|
||||
* @return cJSON object representing the seal event, or NULL on error
|
||||
*/
|
||||
cJSON* nostr_nip59_create_seal(cJSON* rumor, const unsigned char* sender_private_key,
|
||||
const unsigned char* recipient_public_key);
|
||||
const unsigned char* recipient_public_key, long max_delay_sec);
|
||||
|
||||
/**
|
||||
* NIP-59: Create a gift wrap (kind 1059) wrapping a seal
|
||||
*
|
||||
* @param seal The seal event to wrap (cJSON object)
|
||||
* @param recipient_public_key_hex Recipient's public key in hex format
|
||||
* @param max_delay_sec Maximum random timestamp delay in seconds (0 = no randomization)
|
||||
* @return cJSON object representing the gift wrap event, or NULL on error
|
||||
*/
|
||||
cJSON* nostr_nip59_create_gift_wrap(cJSON* seal, const char* recipient_public_key_hex);
|
||||
cJSON* nostr_nip59_create_gift_wrap(cJSON* seal, const char* recipient_public_key_hex, long max_delay_sec);
|
||||
|
||||
/**
|
||||
* NIP-59: Unwrap a gift wrap to get the seal
|
||||
|
||||
@@ -72,11 +72,11 @@
|
||||
#define NIP05_DEFAULT_TIMEOUT 10
|
||||
|
||||
// NIP-04 Constants
|
||||
#define NOSTR_NIP04_MAX_PLAINTEXT_SIZE 16777216 // 16MB
|
||||
#define NOSTR_NIP04_MAX_PLAINTEXT_SIZE 1048576 // 1MB
|
||||
#define NOSTR_NIP04_MAX_ENCRYPTED_SIZE 22369621 // ~21.3MB (accounts for base64 overhead + IV)
|
||||
|
||||
// NIP-44 Constants
|
||||
#define NOSTR_NIP44_MAX_PLAINTEXT_SIZE 65536 // 64KB max plaintext (matches crypto header)
|
||||
// NIP-44 Constants
|
||||
#define NOSTR_NIP44_MAX_PLAINTEXT_SIZE 65535 // 64KB - 1 (NIP-44 spec compliant)
|
||||
|
||||
// Forward declaration for cJSON (to avoid requiring cJSON.h in header)
|
||||
struct cJSON;
|
||||
|
||||
@@ -2,10 +2,10 @@
|
||||
#define NOSTR_CORE_H
|
||||
|
||||
// Version information (auto-updated by increment_and_push.sh)
|
||||
#define VERSION "v0.4.6"
|
||||
#define VERSION "v0.4.8"
|
||||
#define VERSION_MAJOR 0
|
||||
#define VERSION_MINOR 4
|
||||
#define VERSION_PATCH 6
|
||||
#define VERSION_PATCH 8
|
||||
|
||||
/*
|
||||
* NOSTR Core Library - Complete API Reference
|
||||
@@ -288,11 +288,23 @@ cJSON* nostr_relay_pool_get_event(
|
||||
int relay_count,
|
||||
cJSON* filter,
|
||||
int timeout_ms);
|
||||
int nostr_relay_pool_publish(
|
||||
// Async publish callback typedef
|
||||
typedef void (*publish_response_callback_t)(
|
||||
const char* relay_url,
|
||||
const char* event_id,
|
||||
int success, // 1 for OK, 0 for rejection
|
||||
const char* message, // Error message if rejected, NULL if success
|
||||
void* user_data
|
||||
);
|
||||
|
||||
// Async publish function (only async version available)
|
||||
int nostr_relay_pool_publish_async(
|
||||
nostr_relay_pool_t* pool,
|
||||
const char** relay_urls,
|
||||
int relay_count,
|
||||
cJSON* event);
|
||||
cJSON* event,
|
||||
publish_response_callback_t callback,
|
||||
void* user_data);
|
||||
|
||||
// Status and statistics functions
|
||||
nostr_pool_relay_status_t nostr_relay_pool_get_relay_status(
|
||||
|
||||
BIN
tests/async_publish_test
Executable file
BIN
tests/async_publish_test
Executable file
Binary file not shown.
93
tests/async_publish_test.c
Normal file
93
tests/async_publish_test.c
Normal file
@@ -0,0 +1,93 @@
|
||||
#define _DEFAULT_SOURCE
|
||||
#include "../nostr_core/nostr_core.h"
|
||||
#include "../cjson/cJSON.h"
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <unistd.h>
|
||||
|
||||
// Test callback function
|
||||
static int callback_count = 0;
|
||||
static int success_count = 0;
|
||||
|
||||
void test_callback(const char* relay_url, const char* event_id,
|
||||
int success, const char* message, void* user_data) {
|
||||
callback_count++;
|
||||
if (success) {
|
||||
success_count++;
|
||||
}
|
||||
|
||||
printf("📡 Callback %d: Relay %s, Event %s, Success: %s\n",
|
||||
callback_count, relay_url, event_id, success ? "YES" : "NO");
|
||||
if (message) {
|
||||
printf(" Message: %s\n", message);
|
||||
}
|
||||
|
||||
// Mark test as complete when we get the expected number of callbacks
|
||||
int* expected_callbacks = (int*)user_data;
|
||||
if (callback_count >= *expected_callbacks) {
|
||||
printf("✅ All callbacks received!\n");
|
||||
}
|
||||
}
|
||||
|
||||
int main() {
|
||||
printf("🧪 Testing Async Publish Functionality\n");
|
||||
printf("=====================================\n");
|
||||
|
||||
// Create pool
|
||||
nostr_relay_pool_t* pool = nostr_relay_pool_create(NULL);
|
||||
if (!pool) {
|
||||
printf("❌ Failed to create pool\n");
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Create a test event
|
||||
cJSON* event = cJSON_CreateObject();
|
||||
cJSON_AddStringToObject(event, "id", "test_event_12345");
|
||||
cJSON_AddNumberToObject(event, "kind", 1);
|
||||
cJSON_AddStringToObject(event, "content", "Test async publish");
|
||||
cJSON_AddNumberToObject(event, "created_at", time(NULL));
|
||||
cJSON_AddStringToObject(event, "pubkey", "test_pubkey");
|
||||
cJSON_AddStringToObject(event, "sig", "test_signature");
|
||||
|
||||
// Test with non-existent relays (should trigger connection failure callbacks)
|
||||
const char* test_relays[] = {
|
||||
"ws://nonexistent1.example.com",
|
||||
"ws://nonexistent2.example.com"
|
||||
};
|
||||
int expected_callbacks = 2;
|
||||
|
||||
printf("🚀 Testing async publish with connection failure callbacks...\n");
|
||||
|
||||
// Call async publish
|
||||
int sent_count = nostr_relay_pool_publish_async(
|
||||
pool, test_relays, 2, event, test_callback, &expected_callbacks);
|
||||
|
||||
printf("📊 Sent to %d relays\n", sent_count);
|
||||
|
||||
// Wait a bit for callbacks (connection failures should be immediate)
|
||||
printf("⏳ Waiting for callbacks...\n");
|
||||
for (int i = 0; i < 10 && callback_count < expected_callbacks; i++) {
|
||||
nostr_relay_pool_poll(pool, 100);
|
||||
usleep(100000); // 100ms
|
||||
}
|
||||
|
||||
printf("\n📈 Results:\n");
|
||||
printf(" Callbacks received: %d/%d\n", callback_count, expected_callbacks);
|
||||
printf(" Successful publishes: %d\n", success_count);
|
||||
|
||||
// Test backward compatibility with synchronous version
|
||||
printf("\n🔄 Testing backward compatibility (sync version)...\n");
|
||||
int sync_result = nostr_relay_pool_publish_async(pool, test_relays, 2, event, NULL, NULL);
|
||||
printf(" Sync publish result: %d successful publishes\n", sync_result);
|
||||
|
||||
// Cleanup
|
||||
cJSON_Delete(event);
|
||||
nostr_relay_pool_destroy(pool);
|
||||
|
||||
printf("\n✅ Async publish test completed!\n");
|
||||
printf(" - Async callbacks: %s\n", callback_count >= expected_callbacks ? "PASS" : "FAIL");
|
||||
printf(" - Backward compatibility: %s\n", sync_result >= 0 ? "PASS" : "FAIL");
|
||||
|
||||
return (callback_count >= expected_callbacks && sync_result >= 0) ? 0 : 1;
|
||||
}
|
||||
BIN
tests/backward_compat_test
Executable file
BIN
tests/backward_compat_test
Executable file
Binary file not shown.
49
tests/backward_compat_test.c
Normal file
49
tests/backward_compat_test.c
Normal file
@@ -0,0 +1,49 @@
|
||||
#define _DEFAULT_SOURCE
|
||||
#include "../nostr_core/nostr_core.h"
|
||||
#include "../cjson/cJSON.h"
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <unistd.h>
|
||||
|
||||
int main() {
|
||||
printf("🧪 Backward Compatibility Test\n");
|
||||
printf("===============================\n");
|
||||
|
||||
// Create pool
|
||||
nostr_relay_pool_t* pool = nostr_relay_pool_create(NULL);
|
||||
if (!pool) {
|
||||
printf("❌ Failed to create pool\n");
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Create a test event
|
||||
cJSON* event = cJSON_CreateObject();
|
||||
cJSON_AddStringToObject(event, "id", "test_event_sync");
|
||||
cJSON_AddNumberToObject(event, "kind", 1);
|
||||
cJSON_AddStringToObject(event, "content", "Test sync publish");
|
||||
cJSON_AddNumberToObject(event, "created_at", time(NULL));
|
||||
cJSON_AddStringToObject(event, "pubkey", "test_pubkey");
|
||||
cJSON_AddStringToObject(event, "sig", "test_signature");
|
||||
|
||||
// Test with non-existent relay (should return 0 successful publishes)
|
||||
const char* test_relays[] = {"ws://nonexistent.example.com"};
|
||||
|
||||
printf("🚀 Testing synchronous publish (backward compatibility)...\n");
|
||||
|
||||
// Call synchronous publish (old API)
|
||||
int result = nostr_relay_pool_publish_async(pool, test_relays, 1, event, NULL, NULL);
|
||||
|
||||
printf("📊 Synchronous publish result: %d successful publishes\n", result);
|
||||
|
||||
// Cleanup
|
||||
cJSON_Delete(event);
|
||||
nostr_relay_pool_destroy(pool);
|
||||
|
||||
printf("\n✅ Backward compatibility test completed!\n");
|
||||
printf(" Expected: 0 successful publishes (connection failure)\n");
|
||||
printf(" Actual: %d successful publishes\n", result);
|
||||
printf(" Result: %s\n", result == 0 ? "PASS" : "FAIL");
|
||||
|
||||
return result == 0 ? 0 : 1;
|
||||
}
|
||||
BIN
tests/nip04_test
BIN
tests/nip04_test
Binary file not shown.
@@ -671,8 +671,8 @@ int test_vector_7_10kb_payload(void) {
|
||||
printf("Last 80 chars: \"...%.80s\"\n", encrypted + encrypted_len - 80);
|
||||
printf("\n");
|
||||
|
||||
// Test decryption with our ciphertext
|
||||
char* decrypted = malloc(NOSTR_NIP04_MAX_PLAINTEXT_SIZE);
|
||||
// Test decryption with our ciphertext - allocate larger buffer for safety
|
||||
char* decrypted = malloc(NOSTR_NIP04_MAX_PLAINTEXT_SIZE + 1024); // 1MB + 1KB extra
|
||||
if (!decrypted) {
|
||||
printf("❌ MEMORY ALLOCATION FAILED for decrypted buffer\n");
|
||||
free(large_plaintext);
|
||||
@@ -680,7 +680,7 @@ int test_vector_7_10kb_payload(void) {
|
||||
return 0;
|
||||
}
|
||||
printf("Testing decryption of 1MB ciphertext (Bob decrypts from Alice)...\n");
|
||||
result = nostr_nip04_decrypt(sk2, pk1, encrypted, decrypted, NOSTR_NIP04_MAX_PLAINTEXT_SIZE);
|
||||
result = nostr_nip04_decrypt(sk2, pk1, encrypted, decrypted, NOSTR_NIP04_MAX_PLAINTEXT_SIZE + 1024);
|
||||
|
||||
if (result != NOSTR_SUCCESS) {
|
||||
printf("❌ 1MB DECRYPTION FAILED: %s\n", nostr_strerror(result));
|
||||
|
||||
BIN
tests/nip17_test
BIN
tests/nip17_test
Binary file not shown.
BIN
tests/nip44_test
BIN
tests/nip44_test
Binary file not shown.
@@ -20,32 +20,7 @@ typedef struct {
|
||||
const char* expected_encrypted; // Optional - for known test vectors
|
||||
} nip44_test_vector_t;
|
||||
|
||||
// Known decryption-only test vectors from nostr-tools (for cross-compatibility testing)
|
||||
// Note: NIP-44 encryption is non-deterministic - ciphertext varies each time
|
||||
// These vectors test our ability to decrypt known good ciphertext from reference implementations
|
||||
static nip44_test_vector_t decryption_test_vectors[] = {
|
||||
{
|
||||
"Decryption test: single char 'a'",
|
||||
"0000000000000000000000000000000000000000000000000000000000000001", // sec1
|
||||
"0000000000000000000000000000000000000000000000000000000000000002", // sec2
|
||||
"a",
|
||||
"AgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABee0G5VSK0/9YypIObAtDKfYEAjD35uVkHyB0F4DwrcNaCXlCWZKaArsGrY6M9wnuTMxWfp1RTN9Xga8no+kF5Vsb"
|
||||
},
|
||||
{
|
||||
"Decryption test: emoji",
|
||||
"0000000000000000000000000000000000000000000000000000000000000002", // sec1
|
||||
"0000000000000000000000000000000000000000000000000000000000000001", // sec2
|
||||
"🍕🫃",
|
||||
"AvAAAAAAAAAAAAAAAAAAAPAAAAAAAAAAAAAAAAAAAAAPSKSK6is9ngkX2+cSq85Th16oRTISAOfhStnixqZziKMDvB0QQzgFZdjLTPicCJaV8nDITO+QfaQ61+KbWQIOO2Yj"
|
||||
},
|
||||
{
|
||||
"Decryption test: wide unicode",
|
||||
"5c0c523f52a5b6fad39ed2403092df8cebc36318b39383bca6c00808626fab3a", // sec1
|
||||
"4b22aa260e4acb7021e32f38a6cdf4b673c6a277755bfce287e370c924dc936d", // sec2
|
||||
"表ポあA鷗ŒéB逍Üߪąñ丂㐀𠀀",
|
||||
"ArY1I2xC2yDwIbuNHN/1ynXdGgzHLqdCrXUPMwELJPc7s7JqlCMJBAIIjfkpHReBPXeoMCyuClwgbT419jUWU1PwaNl4FEQYKCDKVJz+97Mp3K+Q2YGa77B6gpxB/lr1QgoqpDf7wDVrDmOqGoiPjWDqy8KzLueKDcm9BVP8xeTJIxs="
|
||||
}
|
||||
};
|
||||
// Additional test vectors for edge cases (converted to round-trip tests with new 32-bit padding)
|
||||
|
||||
// Round-trip test vectors with proper key pairs
|
||||
static nip44_test_vector_t test_vectors[] = {
|
||||
@@ -69,6 +44,13 @@ static nip44_test_vector_t test_vectors[] = {
|
||||
"4444444444444444444444444444444444444444444444444444444444444444",
|
||||
"",
|
||||
NULL
|
||||
},
|
||||
{
|
||||
"64KB payload test",
|
||||
"91ba716fa9e7ea2fcbad360cf4f8e0d312f73984da63d90f524ad61a6a1e7dbe", // Same keys as basic test
|
||||
"96f6fa197aa07477ab88f6981118466ae3a982faab8ad5db9d5426870c73d220",
|
||||
NULL, // Will be generated dynamically
|
||||
NULL
|
||||
}
|
||||
};
|
||||
|
||||
@@ -86,76 +68,144 @@ static int hex_to_bytes(const char* hex, unsigned char* bytes, size_t len) {
|
||||
|
||||
static int test_nip44_round_trip(const nip44_test_vector_t* tv) {
|
||||
printf("Test: %s\n", tv->name);
|
||||
|
||||
|
||||
// Parse keys - both private keys
|
||||
unsigned char sender_private_key[32];
|
||||
unsigned char recipient_private_key[32];
|
||||
|
||||
|
||||
if (hex_to_bytes(tv->sender_private_key_hex, sender_private_key, 32) != 0) {
|
||||
printf(" FAIL: Failed to parse sender private key\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
||||
if (hex_to_bytes(tv->recipient_private_key_hex, recipient_private_key, 32) != 0) {
|
||||
printf(" FAIL: Failed to parse recipient private key\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
||||
// Generate the public keys from the private keys
|
||||
unsigned char sender_public_key[32];
|
||||
unsigned char recipient_public_key[32];
|
||||
|
||||
|
||||
if (nostr_ec_public_key_from_private_key(sender_private_key, sender_public_key) != 0) {
|
||||
printf(" FAIL: Failed to derive sender public key\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
||||
if (nostr_ec_public_key_from_private_key(recipient_private_key, recipient_public_key) != 0) {
|
||||
printf(" FAIL: Failed to derive recipient public key\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
// Test encryption
|
||||
char encrypted[8192];
|
||||
int encrypt_result = nostr_nip44_encrypt(
|
||||
sender_private_key,
|
||||
recipient_public_key,
|
||||
tv->plaintext,
|
||||
encrypted,
|
||||
sizeof(encrypted)
|
||||
);
|
||||
|
||||
if (encrypt_result != NOSTR_SUCCESS) {
|
||||
printf(" FAIL: Encryption - Expected: %d, Actual: %d\n", NOSTR_SUCCESS, encrypt_result);
|
||||
|
||||
// Special handling for large payload tests
|
||||
char* test_plaintext;
|
||||
if (strcmp(tv->name, "64KB payload test") == 0) {
|
||||
// Generate exactly 64KB (65,535 bytes) of predictable content - max NIP-44 size
|
||||
const size_t payload_size = 65535;
|
||||
test_plaintext = malloc(payload_size + 1);
|
||||
if (!test_plaintext) {
|
||||
printf(" FAIL: Memory allocation failed for 64KB test payload\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
// Fill with a predictable pattern: "ABCDEFGH01234567" repeated
|
||||
const char* pattern = "ABCDEFGH01234567"; // 16 bytes
|
||||
const size_t pattern_len = 16;
|
||||
|
||||
for (size_t i = 0; i < payload_size; i += pattern_len) {
|
||||
size_t copy_len = (i + pattern_len <= payload_size) ? pattern_len : payload_size - i;
|
||||
memcpy(test_plaintext + i, pattern, copy_len);
|
||||
}
|
||||
test_plaintext[payload_size] = '\0';
|
||||
|
||||
printf(" Generated 64KB test payload (%zu bytes)\n", payload_size);
|
||||
printf(" Pattern: \"%s\" repeated\n", pattern);
|
||||
printf(" First 64 chars: \"%.64s...\"\n", test_plaintext);
|
||||
printf(" Last 64 chars: \"...%.64s\"\n", test_plaintext + payload_size - 64);
|
||||
} else {
|
||||
test_plaintext = (char*)tv->plaintext;
|
||||
}
|
||||
|
||||
// Debug: Check plaintext length
|
||||
size_t plaintext_len = strlen(test_plaintext);
|
||||
printf(" Plaintext length: %zu bytes\n", plaintext_len);
|
||||
printf(" Output buffer size: %zu bytes\n", (size_t)10485760);
|
||||
|
||||
// Test encryption - use larger buffer for 1MB+ payloads (10MB for NIP-44 overhead)
|
||||
char* encrypted = malloc(10485760); // 10MB buffer for large payloads
|
||||
if (!encrypted) {
|
||||
printf(" FAIL: Memory allocation failed for encrypted buffer\n");
|
||||
if (strcmp(tv->name, "0.5MB payload test") == 0) free(test_plaintext);
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
||||
// For large payloads, use _with_nonce to avoid random generation issues
|
||||
unsigned char fixed_nonce[32] = {0};
|
||||
int encrypt_result = nostr_nip44_encrypt_with_nonce(
|
||||
sender_private_key,
|
||||
recipient_public_key,
|
||||
test_plaintext,
|
||||
fixed_nonce,
|
||||
encrypted,
|
||||
10485760
|
||||
);
|
||||
|
||||
if (encrypt_result != NOSTR_SUCCESS) {
|
||||
printf(" FAIL: Encryption - Expected: %d, Actual: %d\n", NOSTR_SUCCESS, encrypt_result);
|
||||
if (strcmp(tv->name, "1MB payload test") == 0) free(test_plaintext);
|
||||
free(encrypted);
|
||||
return -1;
|
||||
}
|
||||
|
||||
// Test decryption - use recipient private key + sender public key
|
||||
char decrypted[8192];
|
||||
char* decrypted = malloc(65536 + 1); // 64KB + 1 for null terminator
|
||||
if (!decrypted) {
|
||||
printf(" FAIL: Memory allocation failed for decrypted buffer\n");
|
||||
if (strcmp(tv->name, "64KB payload test") == 0) free(test_plaintext);
|
||||
free(encrypted);
|
||||
return -1;
|
||||
}
|
||||
int decrypt_result = nostr_nip44_decrypt(
|
||||
recipient_private_key,
|
||||
sender_public_key,
|
||||
encrypted,
|
||||
decrypted,
|
||||
sizeof(decrypted)
|
||||
65536 + 1
|
||||
);
|
||||
|
||||
|
||||
if (decrypt_result != NOSTR_SUCCESS) {
|
||||
printf(" FAIL: Decryption - Expected: %d, Actual: %d\n", NOSTR_SUCCESS, decrypt_result);
|
||||
if (strcmp(tv->name, "1MB payload test") == 0) free(test_plaintext);
|
||||
free(encrypted);
|
||||
free(decrypted);
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
||||
// Verify round-trip
|
||||
if (strcmp(tv->plaintext, decrypted) != 0) {
|
||||
if (strcmp(test_plaintext, decrypted) != 0) {
|
||||
printf(" FAIL: Round-trip mismatch\n");
|
||||
printf(" Expected: \"%s\"\n", tv->plaintext);
|
||||
printf(" Expected: \"%s\"\n", test_plaintext);
|
||||
printf(" Actual: \"%s\"\n", decrypted);
|
||||
if (strcmp(tv->name, "1MB payload test") == 0) free(test_plaintext);
|
||||
free(encrypted);
|
||||
free(decrypted);
|
||||
return -1;
|
||||
}
|
||||
|
||||
printf(" PASS: Expected: \"%s\", Actual: \"%s\"\n", tv->plaintext, decrypted);
|
||||
printf(" Encrypted output: %s\n", encrypted);
|
||||
|
||||
|
||||
if (strcmp(tv->name, "64KB payload test") == 0) {
|
||||
printf(" ✅ 64KB payload round-trip: PASS\n");
|
||||
printf(" ✅ Content verification: All %zu bytes match perfectly!\n", strlen(test_plaintext));
|
||||
printf(" Encrypted length: %zu bytes\n", strlen(encrypted));
|
||||
printf(" 🎉 64KB NIP-44 STRESS TEST COMPLETED SUCCESSFULLY! 🎉\n");
|
||||
} else {
|
||||
printf(" PASS: Expected: \"%s\", Actual: \"%s\"\n", test_plaintext, decrypted);
|
||||
printf(" Encrypted output: %s\n", encrypted);
|
||||
}
|
||||
|
||||
if (strcmp(tv->name, "64KB payload test") == 0) free(test_plaintext);
|
||||
free(encrypted);
|
||||
free(decrypted);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@@ -215,59 +265,6 @@ static int test_nip44_error_conditions() {
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int test_nip44_decryption_vector(const nip44_test_vector_t* tv) {
|
||||
printf("Test: %s\n", tv->name);
|
||||
|
||||
// Parse keys
|
||||
unsigned char sender_private_key[32];
|
||||
unsigned char recipient_private_key[32];
|
||||
|
||||
if (hex_to_bytes(tv->sender_private_key_hex, sender_private_key, 32) != 0) {
|
||||
printf(" FAIL: Failed to parse sender private key\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (hex_to_bytes(tv->recipient_private_key_hex, recipient_private_key, 32) != 0) {
|
||||
printf(" FAIL: Failed to parse recipient private key\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
// Generate the public keys from the private keys
|
||||
unsigned char sender_public_key[32];
|
||||
|
||||
if (nostr_ec_public_key_from_private_key(sender_private_key, sender_public_key) != 0) {
|
||||
printf(" FAIL: Failed to derive sender public key\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
// Test decryption of known vector
|
||||
char decrypted[8192];
|
||||
int decrypt_result = nostr_nip44_decrypt(
|
||||
recipient_private_key,
|
||||
sender_public_key,
|
||||
tv->expected_encrypted,
|
||||
decrypted,
|
||||
sizeof(decrypted)
|
||||
);
|
||||
|
||||
if (decrypt_result != NOSTR_SUCCESS) {
|
||||
printf(" FAIL: Decryption - Expected: %d, Actual: %d\n", NOSTR_SUCCESS, decrypt_result);
|
||||
printf(" Input payload: %s\n", tv->expected_encrypted);
|
||||
return -1;
|
||||
}
|
||||
|
||||
// Verify decrypted plaintext matches expected
|
||||
if (strcmp(tv->plaintext, decrypted) != 0) {
|
||||
printf(" FAIL: Plaintext mismatch\n");
|
||||
printf(" Expected: \"%s\"\n", tv->plaintext);
|
||||
printf(" Actual: \"%s\"\n", decrypted);
|
||||
return -1;
|
||||
}
|
||||
|
||||
printf(" PASS: Expected: \"%s\", Actual: \"%s\"\n", tv->plaintext, decrypted);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int test_nip44_encryption_variability() {
|
||||
printf("Test: NIP-44 encryption variability (non-deterministic)\n");
|
||||
@@ -287,11 +284,20 @@ static int test_nip44_encryption_variability() {
|
||||
}
|
||||
|
||||
// Encrypt the same message multiple times
|
||||
char encrypted1[8192], encrypted2[8192], encrypted3[8192];
|
||||
char* encrypted1 = malloc(2097152); // 2MB buffer
|
||||
char* encrypted2 = malloc(2097152);
|
||||
char* encrypted3 = malloc(2097152);
|
||||
if (!encrypted1 || !encrypted2 || !encrypted3) {
|
||||
printf(" FAIL: Memory allocation failed for encrypted buffers\n");
|
||||
free(encrypted1);
|
||||
free(encrypted2);
|
||||
free(encrypted3);
|
||||
return -1;
|
||||
}
|
||||
|
||||
int result1 = nostr_nip44_encrypt(sender_key, recipient_pubkey, test_message, encrypted1, sizeof(encrypted1));
|
||||
int result2 = nostr_nip44_encrypt(sender_key, recipient_pubkey, test_message, encrypted2, sizeof(encrypted2));
|
||||
int result3 = nostr_nip44_encrypt(sender_key, recipient_pubkey, test_message, encrypted3, sizeof(encrypted3));
|
||||
int result1 = nostr_nip44_encrypt(sender_key, recipient_pubkey, test_message, encrypted1, 2097152);
|
||||
int result2 = nostr_nip44_encrypt(sender_key, recipient_pubkey, test_message, encrypted2, 2097152);
|
||||
int result3 = nostr_nip44_encrypt(sender_key, recipient_pubkey, test_message, encrypted3, 2097152);
|
||||
|
||||
if (result1 != NOSTR_SUCCESS || result2 != NOSTR_SUCCESS || result3 != NOSTR_SUCCESS) {
|
||||
printf(" FAIL: Encryption failed - Results: %d, %d, %d\n", result1, result2, result3);
|
||||
@@ -304,6 +310,9 @@ static int test_nip44_encryption_variability() {
|
||||
printf(" Encryption 1: %.50s...\n", encrypted1);
|
||||
printf(" Encryption 2: %.50s...\n", encrypted2);
|
||||
printf(" Encryption 3: %.50s...\n", encrypted3);
|
||||
free(encrypted1);
|
||||
free(encrypted2);
|
||||
free(encrypted3);
|
||||
return -1;
|
||||
}
|
||||
|
||||
@@ -314,11 +323,23 @@ static int test_nip44_encryption_variability() {
|
||||
return -1;
|
||||
}
|
||||
|
||||
char decrypted1[8192], decrypted2[8192], decrypted3[8192];
|
||||
|
||||
int decrypt1 = nostr_nip44_decrypt(recipient_key, sender_pubkey, encrypted1, decrypted1, sizeof(decrypted1));
|
||||
int decrypt2 = nostr_nip44_decrypt(recipient_key, sender_pubkey, encrypted2, decrypted2, sizeof(decrypted2));
|
||||
int decrypt3 = nostr_nip44_decrypt(recipient_key, sender_pubkey, encrypted3, decrypted3, sizeof(decrypted3));
|
||||
char* decrypted1 = malloc(1048576 + 1);
|
||||
char* decrypted2 = malloc(1048576 + 1);
|
||||
char* decrypted3 = malloc(1048576 + 1);
|
||||
if (!decrypted1 || !decrypted2 || !decrypted3) {
|
||||
printf(" FAIL: Memory allocation failed for decrypted buffers\n");
|
||||
free(encrypted1);
|
||||
free(encrypted2);
|
||||
free(encrypted3);
|
||||
free(decrypted1);
|
||||
free(decrypted2);
|
||||
free(decrypted3);
|
||||
return -1;
|
||||
}
|
||||
|
||||
int decrypt1 = nostr_nip44_decrypt(recipient_key, sender_pubkey, encrypted1, decrypted1, 1048576 + 1);
|
||||
int decrypt2 = nostr_nip44_decrypt(recipient_key, sender_pubkey, encrypted2, decrypted2, 1048576 + 1);
|
||||
int decrypt3 = nostr_nip44_decrypt(recipient_key, sender_pubkey, encrypted3, decrypted3, 1048576 + 1);
|
||||
|
||||
if (decrypt1 != NOSTR_SUCCESS || decrypt2 != NOSTR_SUCCESS || decrypt3 != NOSTR_SUCCESS) {
|
||||
printf(" FAIL: Decryption failed - Results: %d, %d, %d\n", decrypt1, decrypt2, decrypt3);
|
||||
@@ -331,12 +352,25 @@ static int test_nip44_encryption_variability() {
|
||||
printf(" Decrypted1: \"%s\"\n", decrypted1);
|
||||
printf(" Decrypted2: \"%s\"\n", decrypted2);
|
||||
printf(" Decrypted3: \"%s\"\n", decrypted3);
|
||||
free(encrypted1);
|
||||
free(encrypted2);
|
||||
free(encrypted3);
|
||||
free(decrypted1);
|
||||
free(decrypted2);
|
||||
free(decrypted3);
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
||||
printf(" PASS: All encryptions different, all decrypt to: \"%s\"\n", test_message);
|
||||
printf(" Sample ciphertext lengths: %zu, %zu, %zu bytes\n", strlen(encrypted1), strlen(encrypted2), strlen(encrypted3));
|
||||
|
||||
|
||||
free(encrypted1);
|
||||
free(encrypted2);
|
||||
free(encrypted3);
|
||||
free(decrypted1);
|
||||
free(decrypted2);
|
||||
free(decrypted3);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@@ -365,12 +399,37 @@ int main() {
|
||||
printf("\n");
|
||||
}
|
||||
|
||||
// Test decryption vectors (cross-compatibility)
|
||||
size_t num_decryption_vectors = sizeof(decryption_test_vectors) / sizeof(decryption_test_vectors[0]);
|
||||
for (size_t i = 0; i < num_decryption_vectors; i++) {
|
||||
// Additional edge case tests (converted to round-trip tests with new 32-bit padding)
|
||||
// These test the same plaintexts as the old decryption vectors but with our new format
|
||||
static nip44_test_vector_t edge_case_test_vectors[] = {
|
||||
{
|
||||
"Edge case: single char 'a'",
|
||||
"0000000000000000000000000000000000000000000000000000000000000001", // sec1
|
||||
"0000000000000000000000000000000000000000000000000000000000000002", // sec2
|
||||
"a",
|
||||
NULL
|
||||
},
|
||||
{
|
||||
"Edge case: emoji",
|
||||
"0000000000000000000000000000000000000000000000000000000000000002", // sec1
|
||||
"0000000000000000000000000000000000000000000000000000000000000001", // sec2
|
||||
"🍕🫃",
|
||||
NULL
|
||||
},
|
||||
{
|
||||
"Edge case: wide unicode",
|
||||
"5c0c523f52a5b6fad39ed2403092df8cebc36318b39383bca6c00808626fab3a", // sec1
|
||||
"4b22aa260e4acb7021e32f38a6cdf4b673c6a277755bfce287e370c924dc936d", // sec2
|
||||
"表ポあA鷗ŒéB逍Üߪąñ丂㐀𠀀",
|
||||
NULL
|
||||
}
|
||||
};
|
||||
|
||||
size_t num_edge_case_vectors = sizeof(edge_case_test_vectors) / sizeof(edge_case_test_vectors[0]);
|
||||
for (size_t i = 0; i < num_edge_case_vectors; i++) {
|
||||
total_tests++;
|
||||
printf("Test #%d\n", total_tests);
|
||||
if (test_nip44_decryption_vector(&decryption_test_vectors[i]) == 0) {
|
||||
if (test_nip44_round_trip(&edge_case_test_vectors[i]) == 0) {
|
||||
passed_tests++;
|
||||
}
|
||||
printf("\n");
|
||||
|
||||
Binary file not shown.
BIN
tests/simple_async_test
Executable file
BIN
tests/simple_async_test
Executable file
Binary file not shown.
73
tests/simple_async_test.c
Normal file
73
tests/simple_async_test.c
Normal file
@@ -0,0 +1,73 @@
|
||||
#define _DEFAULT_SOURCE
|
||||
#include "../nostr_core/nostr_core.h"
|
||||
#include "../cjson/cJSON.h"
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <unistd.h>
|
||||
|
||||
// Test callback function
|
||||
static int callback_count = 0;
|
||||
|
||||
void test_callback(const char* relay_url, const char* event_id,
|
||||
int success, const char* message, void* user_data) {
|
||||
(void)event_id; // Suppress unused parameter warning
|
||||
(void)user_data; // Suppress unused parameter warning
|
||||
|
||||
callback_count++;
|
||||
printf("📡 Callback %d: Relay %s, Success: %s\n",
|
||||
callback_count, relay_url, success ? "YES" : "NO");
|
||||
if (message) {
|
||||
printf(" Message: %s\n", message);
|
||||
}
|
||||
}
|
||||
|
||||
int main() {
|
||||
printf("🧪 Simple Async Publish Test\n");
|
||||
printf("============================\n");
|
||||
|
||||
// Create pool
|
||||
nostr_relay_pool_t* pool = nostr_relay_pool_create(NULL);
|
||||
if (!pool) {
|
||||
printf("❌ Failed to create pool\n");
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Create a test event
|
||||
cJSON* event = cJSON_CreateObject();
|
||||
cJSON_AddStringToObject(event, "id", "test_event_simple");
|
||||
cJSON_AddNumberToObject(event, "kind", 1);
|
||||
cJSON_AddStringToObject(event, "content", "Test async publish");
|
||||
cJSON_AddNumberToObject(event, "created_at", time(NULL));
|
||||
cJSON_AddStringToObject(event, "pubkey", "test_pubkey");
|
||||
cJSON_AddStringToObject(event, "sig", "test_signature");
|
||||
|
||||
// Test with non-existent relay (should trigger connection failure callback)
|
||||
const char* test_relays[] = {"ws://nonexistent.example.com"};
|
||||
|
||||
printf("🚀 Testing async publish...\n");
|
||||
|
||||
// Call async publish
|
||||
int sent_count = nostr_relay_pool_publish_async(
|
||||
pool, test_relays, 1, event, test_callback, NULL);
|
||||
|
||||
printf("📊 Sent to %d relays\n", sent_count);
|
||||
|
||||
// Wait a bit for callback
|
||||
printf("⏳ Waiting for callback...\n");
|
||||
for (int i = 0; i < 5 && callback_count == 0; i++) {
|
||||
nostr_relay_pool_poll(pool, 100);
|
||||
usleep(100000); // 100ms
|
||||
}
|
||||
|
||||
printf("\n📈 Results:\n");
|
||||
printf(" Callbacks received: %d\n", callback_count);
|
||||
|
||||
// Cleanup
|
||||
cJSON_Delete(event);
|
||||
nostr_relay_pool_destroy(pool);
|
||||
|
||||
printf("\n✅ Simple async test completed!\n");
|
||||
|
||||
return callback_count > 0 ? 0 : 1;
|
||||
}
|
||||
Reference in New Issue
Block a user