Skip to content
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

IPNS-over-PubSub as an Independent Transport #218

Merged
merged 6 commits into from
Apr 30, 2020

Conversation

aschmahmann
Copy link
Contributor

IPNS over PubSub does not currently operate as an independent transport since it only receives updates, and has no way of getting existing published records. Additionally, even the current IPNS over PubSub does not have a spec.

This PR is meant to develop an IPNS over PubSub spec based on some of the ongoing work at ipfs/kubo#6447.

This spec is also dependent on a libp2p spec for the Fetch protocol, which will be posted here as soon as a PR is up.

Please comment, make suggestions, and help make this spec better and easier to work with 😄

^ @hugomrdias @lidel would appreciate whether this spec looks complete enough that you wouldn't have to look at the Go code to implement this in JS.

Copy link
Member

@lidel lidel left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@hugomrdias is OOO this week, but I made a quick pass focused on readability:

naming/pubsub.md Outdated Show resolved Hide resolved
naming/pubsub.md Outdated Show resolved Hide resolved
naming/pubsub.md Outdated Show resolved Hide resolved
naming/pubsub.md Outdated Show resolved Hide resolved
naming/pubsub.md Show resolved Hide resolved
naming/pubsub.md Show resolved Hide resolved
naming/pubsub.md Show resolved Hide resolved
naming/pubsub.md Show resolved Hide resolved
naming/pubsub.md Show resolved Hide resolved
@aschmahmann
Copy link
Contributor Author

@lidel you think we're ready to merge this?

Copy link
Member

@lidel lidel left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(resolved some questions to make it easier for others to review)

Small question about base64 below, apart from that LGTM.
Would like to have @Stebalien or someone familiar with historical ipfs/spec process to eyeball this.

naming/pubsub.md Show resolved Hide resolved

For a given IPNS local record key described in the IPNS Specification the PubSub topic is:

**Topic format:** `/record/base64url-unpadded(key)`
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I see ipfs/specs/naming#local-record uses /ipns/base32(<HASH>).
We could use the same naming convention in all places, if possible.

What would be less work, change this to /ipns/base32(key) or the other way around?

Copy link
Contributor Author

@aschmahmann aschmahmann Sep 5, 2019

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good point about the inconsistency, there's two issues here:

  1. I think that area of the IPNS spec could use some clarification. Here, we're actually referring to the ipfs/specs/naming#routing-record /ipns/BINARY_ID.
  2. This spec is currently using /record instead of /ipns. Should we make it use /ipns? If starting with /ipns is not necessary for routing records across multiple routers then we should probably just remove the section on routing records.

What do you think?

Edit: Just an update/clarification that IPNS over PubSub currently uses /record as the topic. It's possible this was an oversight, is this something we should change going forward?

Copy link
Contributor Author

@aschmahmann aschmahmann Sep 5, 2019

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I asked @Stebalien what he thought about this and his understanding of the linked IPNS is that:

local-record = what we put in a local datastore = /ipns/base32(key) where key = multihash of IPNS public key
routing-record = what the form of the routing record is (BINARY_ID = binary representation of key where key = multihash of IPNS public key

This means our topic format is /record/base64url-unpadded(/ipns/BINARY_ID). The topic format being /record is therefore not actually incompatible with the already described routing-record for IPNS.

However, we should give BINARY_ID a new name in the other spec that makes more sense. One set of words we tend to use is IPNS private key = signing key, IPNS public key = verification key, IPNS key = multihash(IPNS public key). However, we could probably use better words, any suggestions?

Copy link
Member

@lidel lidel Sep 12, 2019

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@aschmahmann
I've been thinking about naming and got some ideas on how to simplify things a bit:

👍 unify popular terms

As you noted, public/private provides good mental model and we should keep it in specs.

What is ambiguous is IPNS key which is just a shortcut/fingerprint that acts as "IPNS identifier".

I suspect something like this would have the least cognitive overhead:

  • IPNS Private Key = signing key
  • IPNS Public Key = verification key
  • IPNS ID / Identifier = multihash(IPNS Public Key)

Renaming BINARY_ID to BINARY_IPNS_ID should do the trick, it match "IPNS ID" nicely.

✍️ Rename pubsub topic, make it a valid IPNS path?

My concerns with /record/:

  • ambiguity during debugging
    • can't guess what is the purpose of a /record/ topic without deserializing base64
  • adding unnecessary noise when doing ipfs pubsub ls
    • can't just open IPNS path without doing additional conversion
      (compare: /record/{base64} vs /ipns/{cid})

What if we simplify this and rename pubsub topic to follow text representation proposed in libp2p/specs#209? Basically /ipns/cid(IPNS ID)

For CIDv1 default text representation would be in Base32:

base32(cid(1, libp2p-key-codec, multihash(IPNS Public Key)))

Key idea here is that when you do ipfs pubsub ls, you get topics that are valid IPNS content paths. User can just copy topic name and IPNS resource will load, without doing any additional conversion (huge UX win).

Thoughts?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

unify popular terms

👍 to that whole section. Those names sound fine to me and are at least not ambiguous anymore.

Rename pubsub topic, make it a valid IPNS path

This bothers me too, especially the ipfs pubsub ls being pretty unhelpful bit.

I think the reason for /record was to indicate that PubSub has this generalized strategy for dealing with records (e.g. the "keep latest record" strategy could be replaced with basically any function that looks like state = merge(state, new record) as described in libp2p/go-libp2p-pubsub-router#36).

I'm a little concerned about using /ipns/base32(cid(1, libp2p-key-codec, multihash(IPNS Public Key))) as a PubSub topic. While libp2p/specs#209 still allows PeerIDs to be searched for or referenced on the network by multihash, this change would make IPNS IDs effectively CIDs since the network level representation in PubSub is the topic name.

A couple options to do something like this might be:

  1. Just make IPNS IDs CIDs
    • this will likely run into the same problems as PeerIDs as CIDs (e.g. bump CID version number and now the topic is different)
  2. Use /ipns/base64(multihash(IPNS Public Key)) (could also be base32 if we wanted)
    • indicates it's an IPNS record, but not super helpful for figuring out which one (since IPNS IDs are not base64(multihash(IPNS Public Key)
    • could combine this with an aliasing function for debugging tools (e.g. ipfs pubsub ls or Wireshark) that processes it into a format we might prefer like /ipns/base32(cid(1, libp2p-key-codec, multihash(IPNS Public Key)))

@Stebalien any thoughts on /record vs /ipns for the PubSub topic?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

local-record = what we put in a local datastore = /ipns/base32(key) where key = multihash of IPNS public key

Turns out this isn't how it works in go. Instead, we just base32 the entire key: base32("/ipns/binary_peer_id).

This spec is currently using /record instead of /ipns. Should we make it use /ipns? If starting with /ipns is not necessary for routing records across multiple routers then we should probably just remove the section on routing records.

We use /record because the go-libp2p-routing-helpers is record type agnostic. It's not specific to IPNS records, it works with all records. We wanted a namespace this special router could own inside the pubsub topic namespace so it wouldn't conflict with other pubsub topics.

Really, IPNS-over-PubSub should be renamed to Records-over-PubSub.

can't guess what is the purpose of a /record/ topic without deserializing base64

The issue here was that record keys are arbitrary binary but pubsub requires utf-8 keys.

adding unnecessary noise when doing ipfs pubsub ls

I agree this kind of sucks but this is one of the reasons we have ipfs name pubsub subs.


My thoughts: leave it as it is, maybe rename this spec to "records over pubsub".

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We could rename the spec as this is really just the spec for go-libp2p-routing-helpers.

However, given that the record agnostic go-libp2p-routing-helpers requires a record validator and implicitly assumes that other subscribers to the same topic have the same validators it might be reasonable for it to take both a protocol prefix and a validator.

The question is, do we get anything out of having this /record namespace since it looks like /ipns might be more helpful?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, we get to reuse the same validator logic, the same record keys, the same everything. We can add support for a new record type to the entire system just by adding the record type to the validator.

@daviddias
Copy link
Member

@aschmahmann I was going to rebase this PR onto master, but found that it got created in your fork. Would you prefer:

  • rebasing yourself
  • creating a new branch from this repo so that I can then rebase it myself?

Also, this is a good opportunity to document the multiple routers of IPNS and not just make the case for PubSub.

@aschmahmann
Copy link
Contributor Author

creating a new branch from this repo so that I can then rebase it myself?

I can create a new branch in this repo.

Also, this is a good opportunity to document the multiple routers of IPNS and not just make the case for PubSub.

Absolutely, also we should make the IPNS spec itself cleaner (e.g. #205). There's also some ideas about what the spec should be for an IPNS address (i.e. /ipns/key) in the presence of multiple routers #198. It's not clear what the best solution should be, but we should document what our current plan is.

Copy link
Member

@Stebalien Stebalien left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You know, all this would be so much simpler if we CIDs as message IDs...

@hsanjuan
Copy link
Contributor

hey @aschmahmann , I am not sure why this is hanging here, should merge?

@aschmahmann
Copy link
Contributor Author

@hsanjuan don't think there's any particular reason, other than I was being a perfectionist and got a bit distracted by other priorities 😬.

The two things I wanted to revise/revisit were:

  1. Renaming the spec as @Stebalien mentioned
  2. Considering whether republishing should be considered optional or not (it should be optional, but due to some PubSub issues it's technically not, although we can likely have very long durations between republish times).

I'll try and fix this up this week/weekend and if it's not done by then let's merge on Monday? You're right that this has been sitting here too long.

@hsanjuan
Copy link
Contributor

hsanjuan commented Apr 1, 2020

Thanks!

/remind me to merge on Monday

@hsanjuan
Copy link
Contributor

hsanjuan commented Apr 1, 2020

ah, reminders only work on issues, not PRs :(

@aschmahmann
Copy link
Contributor Author

aschmahmann commented Apr 30, 2020

@hsanjuan let's merge this before I forget again (I don't have permissions) 😄. Updates will come in a followup PR.

@hsanjuan hsanjuan merged commit 7b2c0ea into ipfs:master Apr 30, 2020
@hsanjuan
Copy link
Contributor

Interesting, only ipfs/docs or ipfs/admin can merge in this repo, but spec writers like you are not in those teams.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants