package namesys import ( "fmt" "time" "code.google.com/p/goprotobuf/proto" ci "github.com/jbenet/go-ipfs/crypto" mdag "github.com/jbenet/go-ipfs/merkledag" "github.com/jbenet/go-ipfs/routing" u "github.com/jbenet/go-ipfs/util" mh "github.com/jbenet/go-multihash" ) // RoutingResolver implements NSResolver for the main IPFS SFS-like naming type RoutingResolver struct { routing routing.IpfsRouting dag *mdag.DAGService } func NewRoutingResolver(route routing.IpfsRouting, dagservice *mdag.DAGService) *RoutingResolver { return &RoutingResolver{ routing: route, dag: dagservice, } } func (r *RoutingResolver) Matches(name string) bool { _, err := mh.FromB58String(name) return err == nil } func (r *RoutingResolver) Resolve(name string) (string, error) { hash, err := mh.FromB58String(name) if err != nil { u.DOut("RoutingResolve: bad input hash: [%s]\n", name) return "", err } // name should be a multihash. if it isn't, error out here. // use the routing system to get the name. // /ipns/ h, err := u.Hash([]byte("/ipns/" + name)) if err != nil { return "", err } ipnsKey := u.Key(h) val, err := r.routing.GetValue(ipnsKey, time.Second*10) if err != nil { u.DOut("RoutingResolve get failed.\n") return "", err } entry := new(IpnsEntry) err = proto.Unmarshal(val, entry) if err != nil { return "", err } // name should be a public key retrievable from ipfs // /ipfs/ key := u.Key(hash) pkval, err := r.routing.GetValue(key, time.Second*10) if err != nil { u.DOut("RoutingResolve PubKey Get failed.\n") return "", err } // get PublicKey from node.Data pk, err := ci.UnmarshalPublicKey(pkval) if err != nil { return "", err } // check sig with pk if ok, err := pk.Verify(entry.GetValue(), entry.GetSignature()); err != nil || !ok { return "", fmt.Errorf("Invalid value. Not signed by PrivateKey corresponding to %v", pk) } // ok sig checks out. this is a valid name. return string(entry.GetValue()), nil }