-
Notifications
You must be signed in to change notification settings - Fork 0
nat/natpmp: add low-level Conn and Message APIs #3
Comments
There's no way in this API for me to deliver messages into the Conn, that's the main missing thing I think. When using a multiplexed socket, the natpmp package can only use that socket for sending, never for receiving. Something else, in the caller's code, is receiving raw bytes and needs some way to tell "do these bytes look like a NAT-PMP message?" When that happens, it needs an API in this package to deliver those bytes, e.g.
Then DeliverByte acts as if those bytes have just been read off the net.PacketConn, and processes them appropriately. For the use of this package on a multiplexed socket, SendMessage and ReceiveMessage aren't really useful as exported types, in that we never expect to poke at NAT-PMP messages ourselves. The Conn also needs some kind of setting so that the rest of the library never reads from the PacketConn directly, or things will break. ParseMessage can be used as the packet discriminator in the demultiplexer, but it's likely to be quite expensive, which is bad since it has to run on every packet received. It'd be nice to have some kind of This is where it gets dicey implementing NAT-PMP as a general purpose library, because you have to co-design the protocol you're multiplexing with so that NAT-PMP is unambiguous compared to the packets in the main protocol. |
I suppose I'm having a hard time understanding: you're saying that NAT-PMP/PCP/etc must be multiplexed with your application protocol on the same socket? Taking a look at https://tools.ietf.org/html/rfc6886#section-3.3, it seems that you can specify your internal TCP/UDP port so the NAT negotations can happen out of band from the application protocol itself. This would allow the NAT negotiations to happen and open the port for the application protocol. Does this not really work in the real world? |
Hm. I'm puzzled. I was convinced that NAT-PMP and PCP both required the mapping request to come from the internal ip:port that the mapping should forward to. But looking again, I see that the target port is freely selectable in both protocols, and only the IP address is fixed (although PCP also has a THIRD_PARTY extension that allows editing mappings on behalf of some other IP). So, I guess everything I said in #1 and here is not applicable. This library can run completely independently of anything else. Sorry! |
It's all good! I was having a hard time discerning if your requirements were one or more of:
I'll keep exploring and see what I can come up with. I do intend to make a type that multiplexes NAT-PMP and PCP as much as possible, but I haven't explored what that might look like yet. |
Per the discussions with @danderson in #1, there is a need to enable sending/receiving NAT-PMP (and later PCP) messages over a single multiplexed UDP socket.
I think the best way to do this is to add a low-level Conn API (inspired by my NDP package) that looks something like so:
The existing Client can make use of this API in a very concise way and keep all the existing serialization and backoff/retry logic. It's unclear to me if any of that logic should live in Conn directly, but I'm leaning toward keeping it out.
For the Tailscale netcheck use case, it'd be easy to probe for NAT-PMP (and later PCP) using Conn.SendMessage, and then messages could be received using a raw Conn.ReadFrom combined with a call to ParseMessage.
Overall I think this approach provides significant flexibility while also allowing a nice low and high-level APIs. Thoughts, @danderson?
The text was updated successfully, but these errors were encountered: