-
Notifications
You must be signed in to change notification settings - Fork 605
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
NIP-88: Notify request spec for relays #901
base: master
Are you sure you want to change the base?
Conversation
I don't like this idea, I think it is too short-sighted. Ties Nostr too much into Bitcoin and Lightning. It's cool for its limited use case, but I don't see it gaining much adoption, even if I liked it. Another concern I have is: will you throw this popup up for every relay that sends a I feel like a Or, if you want a very custom experience, do a normal |
Every different PAY message, yes. This works for all types of relays, even public ones that just want donations. Any payment request any relay wants to send will be displayed. If they annoy the user, it's their customer to loose.
We don't know what types of access relays will be requesting payments for. So, it's better for them to send us a note when we ask for something that is behind a paywall.
We could use NOTICE as well, but we will need to know which ones are DEBUG messages and which NOTICES are to be displayed to the user immediately. Today, the only use of NOTICE is for debugging. A popup for each NOTICE is too much. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I've been wanting to see this for a long time, but now that it comes to it I think I agree with fiatjaf that we should start with a URL rather than an invoice (although NOTICE seems like a bad idea). That way we're not tying ourselves to lightning, but we can also handle other use cases like subscriptions, terms of service, and anything else a relay cares to pack into the page its url points to. The key pain point this solves is flagging within clients that a rely requires admission, which doesn't work at all currently, not facilitating payment, since invoices have to be relayed to wallet software anyway.
88.md
Outdated
] | ||
``` | ||
|
||
Supporting clients SHOULD display a popup or notification to describe the action needed and collect immediate response from the user: `pay` and keep using the relay, `dismiss` and stop using the relay, or `more information` by navigating the user to the URL. Clients SHOULD expect that the same request might be sent multiple times, if the user dismisses or pays the amount, the client SHOULD ignore following requests with the same invoice. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I would leave out the first sentence here. Clients SHOULD handle this in a way that makes sense. A pop-up might not be it.
URL alone does add friction though. My goal was to get going as soon as possible. On-click, NWC payment if possible. The user never leaves the app. Every extra click matters. Also, the URL is there, but it is the exception: "pay this to continue now or look at this URL for terms + other options". |
What if clients just "open" the url and custom protocol handlers can take over from there? So https: would go to a browser, nostr: would go to the user's nostr app, invoice: (or whatever that prefix would be) would open a wallet. Clients can then special case those custom protocols if they want to handle them explicitly. |
We can do lightning:lnbc... I am not sure if the generalization is useful. Maybe we can make the invoice optional and the URL the message's identifier? |
I would be fine with this. To justify the generalization though, it would allow relays to control the UX. If all they want is a payment and want to support a minimal-clicks UX, they can send an invoice and clients can use NWC to pay it (or redirect to a wallet if NWC isn't supported). If they want to offer other payment methods, ask users to create an account or accept a ToS, or provide additional tools for relay configuration or whatever, they can send a vanilla url to a page they own. They could even redirect users to a note using a |
Well, if that is the case, then we can just have a URL and a Kind1-renderable description. So, If there is a lightning invoice in it, it will be shown. If there is a cashu token as a refund, it will be shown as well. Any other links would be clickable. Images can also be present. Then the description should contain the entire text, including what to do if you decide to not continue to pay for the subscription (e.g. removing the relay from the relay list). Then relays can also use it to show a... post of the day, for instance. It is not a PAY-only instruction anymore. We can do a |
I can't tell if you're being sarcastic, but yes haha that sounds fine. Although I would again not call it interrupt_user, since it's not necessary to make it an interruption. Clients could aggregate these in the background and create a digest, automatically handle them, whatever. |
Sorry. No, I wasn't sarcastic :) The issue is that the message is related to whatever the user is trying to use at that moment. For instance, if the user is trying to access Chats and this relay is paid for Chats, the relay won't return anything until the user Pays and the screen will be empty. So, it makes sense to have an immediate interruption of the flow in order to continue the UX the user wants. Putting my sales hat on, an immediate response is a big deal. If these messages just go to notifications then we don't need a spec for it. The relay can just send a Nostr event that will show up in Notifications anyway. The popup at the right time gives the reason to buy. If the user waits, he might not understand why that payment message is even there. |
Interesting point. Why do we need to add this to the spec then? A relay could send a NIP 24 DM from the relay's pubkey specifying payment details (or we could just create a new kind that represents a general-purpose payment request). This is how it's already done in fact. All the UX stuff you're talking about could be supported by clients pretty easily by identifying pubkeys that belong to relays. Latency isn't a problem, because the relay can send the DM using the connection that's already open. |
Sure, as long as the new event kind has the current filter/subscription in it so the app knows which request (or post) triggered it, we can do it as an event kind as well. I don't think we have an official pubkey per relay, do we? We have the owner's key, but not a key that the relay could use to send new notifications anywhere. Clients will have to add a new filter to listen to those events from special keys for each relay being used, but sure. I think this is just simpler. |
Ok, I just changed this to a more generic messaging spec and removed the direct dependency on bitcoin invoices. Let me know what you all think. |
We've been working with Vitor on this and we are happy with the latest version of using NOTIFY instead of PAY and letting the relay decide what messaging to use. I think it is important that clients always show the ws url that sent the NOTIFY when it is displayed, particularly if it includes an invoice. |
So many ideas with this design :) |
I would reserve these notifications for more immediate cases, like pay now to get access to the data you are trying to search for in this screen. Everything else could be a simple new Nostr event that will appear in the notification bar like everything else. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Too inflexible. A notice to users about relay maintenance (which should be shown as a notification), a user needing to pay for access (which should be presented when they add the relay), a user needing to renew their subscription (which should be presented immediately), and other information all get bunched into one message.
Also, invoices should not be created unless the client needs it. I don't get why you are trying to do "server pushes invoices even if the client doesn't need it" for anything related to relay payments, but that one extra round trip is worth the load you are saving on relays.
If you want to send notifications, send notifications via regular Nostr events or DMs. Those things already exist. This spec is not for that. This is explicitly for actions that need to interrupt or capture the user in the moment.
Feel free to create a new encoding to be used inside a Kind 1 that would tell Clients to ping the server, create and pay an invoice for a given amount. Then you can also give people multiple options right in the message. |
Not every client implements DMs. We are going to repeat the same mistake as NIP-50. Someone comes along (nostr.band or nostr.wine), wants to standardize something (search or relay notifications), but wants to avoid changing their implementation (single query for search or sending a notification with the invoice) at the cost of restricting the use cases or resulting in a worse experience for anyone that doesn't want to follow that strict model (search being limited in terms of autocomplete and sorting and specifying params or notifications having varying degrees of importance or multiple choices or the client requesting invoices from the server) |
As usual, you're misinformed and not helpful. We created an implementation for this NIP from scratch so I'm not sure what you are talking about. We certainly aren't forcing this spec to fit something that already exists on our end. Our #1 priority is client interoperability across anything we deploy. We can and will update our integration to fit whatever standard is ultimately agreed upon. We actually spend our time developing our relay not just yelling at the sky. If you don't like the NIP as proposed why don't you suggest some actual changes instead of drawing false conclusions about NIP-50 which has nothing to do with this proposal. |
@Semisol feel free to propose an alternative that is actually implemented in your relays. I am happy to code and test if it's better than this. As you know, I have been asking for something like this for 5 months already. You ended up never proposing anything. @nostr-wine not only proposed, but implemented, shipped and it's working really well. Same for NIP-50, you have not proposed anything workable yet. |
agree this workflow is pretty slick but it makes me concerned about the user experience for non paying users. most paid relays will likely still allow some level of querying kind 1 events, and clients who query a paid relay on an adhoc basis (based on their follows, or a follows set list, or relay hint) will get bombarded with these notifications from relays they don't intend to pay for. long term I don't see the typical user paying for more than 5 relays, but I do anticipate clients querying much more than 5 relays. so imo this has the potential to do more harm than good and will lead to pop up hell. I realize that clients can just choose to opt out, or opt out for all but a user's listed relays (which assumes all the relays a user pays for are used in their relay list), but generally we shouldn't create new message types that could largely be ignored. I think ephemeral events would be much better suited for this use case (this way relays don't have to store notification events for every user and can create them on the fly with any key). create a new ephemeral event kind for relay notifications, and clients can query for that kind if they want to display the relay's notifications to the user. clients can put the user's pubkey in filter #p to indicate which user's notifications should be sent. you could also define different types of notifications with |
Clients can block Notifications from relays that are not on the user's relay list. It's up to them. But if people have a paid relay in their list and are not paying, I think it's fair to blast them with payment requests. They can choose to pay or to remove the relay from their main relay list. If clients are pinging different relays than those the user expects them to ping, they don't need to show such notifications if they don't want to. The NOTIFY works best for relays the user knows and is choosing to use.
Relays do that all the time, though. Most of the "NOTICE" messages are completely ignored.
It's an alternative. But you have to create the infrastructure to make it work. Clients will need to know what is the pubkey used for those messages for each relay and discard everything else. Right now, relays don't have a pubkey for them to message clients. Clients will also need to make sure they don't notify from an old event that was stored in a separate relay and it just happens that that relay decided to send that event. In other words, competing relays can keep old versions of these notifications, and as long as they are sent before a newer event, the user will think it is real. Basically, relays can send a popup hell chain by collecting and rebroadcasting events from other relays. Anyway, we went there and it looked very messy. I am not opposed to it, but I don't know if I mapped out all the threats in that model. |
agree, but its the only message type that is ignored, and it is needed bc we need at least one way to communicate why connection failed, which shouldn't generally be user facing. this wasn't designed to be a user facing message, NOTIFY is.
no that is not what I am proposing. the pubkey of the ephemeral event could be anything and the client doesn't need to know what it is. the client would just send a req with filter { kinds: [relayNotificationKind], #p: [userPubkey] } and the relay would know to create an ephemeral event (with any key) to send to the client. this should require zero extra infrastructure above what is proposed in the NOTIFY workflow. basically a relay notification event should be associated with the relay it comes from, not the pubkey on the event. so the "competing relays can keep old versions of these notifications" doesn't really apply. |
Why do you need to filter, then? The relay can just send it at any point in the regular subscription. It's weird that it is a random key. The relay will need to make sure to not accept these events from anyone else because there is no way for the client to check if this was created at the relay or if it was broadcasted by somebody else. Sounds dangerous and not backwards compatible. Relays that don't implement this variation wouldn't know they have to block a given event kind and will pass the fake notifications down to the client. |
because otherwise there is no specific sub id to associate the event message with. plus relay should only send events in response to reqs, so that it is opt in on client side.
correct, but seems like pretty trivial logic.
agree that's technically possible, but seems like an unrealistic attack, especially if using #p in your filter. takes a lot of effort for very low gain. besides, the nip could recommend that clients only request notifications from relay that support the relay notifications nip (which they could query via nip11) |
After I saw people trying to dupe users with fake replies on the Nostr Wallet Connect, I think everything is possible. But in that case, the client knows which key is supposed to be used and can discard everything else.
I am not really sure if it is very low. Somebody can claim that their subscription to nostr.wine is up and include an invoice to the attacker's wallet. Just create payloads for each user in Nostr and send them all to unsupporting relays. This could be a profitable attack. But if somebody wants to create this variation, I can do a test code as well. |
they would have continuously be sending these for thousands of users to multiple relays (while avoiding being filtered by relay firewall) in the hope that for one specific user it overlaps with the 1-2 sec it takes the client to query for relay notifications (as the sub really shouldn't need to be kept open). sounds like lot of needed resources with low probability of success and low payout. and again, could recommend only querying relays who support this nip. |
Wanted to bump this and say nostr.wine have been using NOTIFY successfully with Amethyst across several of our relays for 4 months now. It's very useful for relays to be able to communicate directly with users for things that require immediate action (most notably payments). We'd love to see other clients implement this NIP. |
This doesn't seem like a well thought out solution:
Are you saying relays must watermark their messages to make sure they never repeat? |
This was more of a protection against poorly implemented relays than anything else. We see relays sending 20 AUTH messages a second, so it's not hard to imagine that some implementations will send similar amounts of NOTIFY. For Amethyst, we only allow one new message per session (go to the background and come back). Repeated messages can play when the user resumes the app unless the user marks to always dismiss this message, then they won't see it anymore. |
This function worked almost as similiar as javascript |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I was not aware of this pull request, it makes sense, I see it as easy to implement, simple, and it helps mitigate (in my opinion) a problem of Nostr, the long-term sustainability of the relays.
I would like to also add that this requires generating LN invoices for each connection, which can be a bottleneck. A standard for requesting a payment invoice should also be included (this could also be additional buttons or a title for the NOTIFY request), instead of it being sent in the NOTIFY where not all clients will render them properly. |
Paid relays often run into friction when renewing subscriptions or when upselling users to the next plan they have available. This PR allows relays to request payments on an as-needed basis.
This has been implemented on inbox.nostr.wine and Amethyst.
Read: https://github.com/vitorpamplona/nips/blob/pay-spec/88.md