package openid import ( "errors" "io" "golang.org/x/net/html" ) func htmlDiscovery(id string, getter httpGetter) (opEndpoint, opLocalID, claimedID string, err error) { resp, err := getter.Get(id, nil) if err != nil { return "", "", "", err } opEndpoint, opLocalID, err = findProviderFromHeadLink(resp.Body) return opEndpoint, opLocalID, resp.Request.URL.String(), err } func findProviderFromHeadLink(input io.Reader) (opEndpoint, opLocalID string, err error) { tokenizer := html.NewTokenizer(input) inHead := false for { tt := tokenizer.Next() switch tt { case html.ErrorToken: // Even if the document is malformed after we found a // valid <link> tag, ignore and let's be happy with our // openid2.provider and potentially openid2.local_id as well. if len(opEndpoint) > 0 { return } return "", "", tokenizer.Err() case html.StartTagToken, html.EndTagToken, html.SelfClosingTagToken: tk := tokenizer.Token() if tk.Data == "head" { if tt == html.StartTagToken { inHead = true } else { if len(opEndpoint) > 0 { return } return "", "", errors.New( "LINK with rel=openid2.provider not found") } } else if inHead && tk.Data == "link" { provider := false localID := false href := "" for _, attr := range tk.Attr { if attr.Key == "rel" { if attr.Val == "openid2.provider" { provider = true } else if attr.Val == "openid2.local_id" { localID = true } } else if attr.Key == "href" { href = attr.Val } } if provider && !localID && len(href) > 0 { opEndpoint = href } else if !provider && localID && len(href) > 0 { opLocalID = href } } } } // At this point we should probably have returned either from // a closing </head> or a tokenizer error (no </head> found). // But just in case. if len(opEndpoint) > 0 { return } return "", "", errors.New("LINK rel=openid2.provider not found") }