| // Copyright 2015 The Vanadium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package caveats |
| |
| import ( |
| "errors" |
| "fmt" |
| "net/http" |
| "strings" |
| "time" |
| |
| "v.io/x/ref/services/identity/internal/templates" |
| |
| "v.io/v23/security" |
| ) |
| |
| type browserCaveatSelector struct { |
| assetsPrefix string |
| blessingName string |
| } |
| |
| // NewBrowserCaveatSelector returns a caveat selector that renders a form in the |
| // to accept user caveat selections. |
| func NewBrowserCaveatSelector(assetsPrefix, blessingName string) CaveatSelector { |
| return &browserCaveatSelector{assetsPrefix, blessingName} |
| } |
| |
| func (s *browserCaveatSelector) Render(blessingExtension, state, redirectURL string, w http.ResponseWriter, r *http.Request) error { |
| tmplargs := struct { |
| Extension string |
| CaveatList []string |
| Macaroon, MacaroonURL, AssetsPrefix, BlessingName string |
| }{blessingExtension, []string{"ExpiryCaveat", "MethodCaveat", "PeerBlessingsCaveat"}, state, redirectURL, s.assetsPrefix, s.blessingName} |
| w.Header().Set("Context-Type", "text/html") |
| if err := templates.SelectCaveats.Execute(w, tmplargs); err != nil { |
| return err |
| } |
| return nil |
| } |
| |
| func (s *browserCaveatSelector) ParseSelections(r *http.Request) (caveats []CaveatInfo, state string, additionalExtension string, err error) { |
| if caveats, err = s.caveats(r); err != nil { |
| return |
| } |
| state = r.FormValue("macaroon") |
| additionalExtension = r.FormValue("blessingExtension") |
| return |
| } |
| |
| func (s *browserCaveatSelector) caveats(r *http.Request) ([]CaveatInfo, error) { |
| if err := r.ParseForm(); err != nil { |
| return nil, err |
| } |
| var caveats []CaveatInfo |
| // Fill in the required caveat. |
| switch required := r.FormValue("requiredCaveat"); required { |
| case "Expiry": |
| expiry, err := newExpiryCaveatInfo(r.FormValue("expiry"), r.FormValue("timezoneOffset")) |
| if err != nil { |
| return nil, fmt.Errorf("failed to create ExpiryCaveat: %v", err) |
| } |
| caveats = append(caveats, expiry) |
| case "Revocation": |
| revocation := newRevocationCaveatInfo() |
| caveats = append(caveats, revocation) |
| default: |
| return nil, fmt.Errorf("%q is not a valid required caveat", required) |
| } |
| if len(caveats) != 1 { |
| return nil, fmt.Errorf("server does not allow for un-restricted blessings") |
| } |
| |
| // And find any additional ones |
| for i, cavName := range r.Form["caveat"] { |
| var err error |
| var caveat CaveatInfo |
| switch cavName { |
| case "ExpiryCaveat": |
| caveat, err = newExpiryCaveatInfo(r.Form[cavName][i], r.FormValue("timezoneOffset")) |
| case "MethodCaveat": |
| caveat, err = newMethodCaveatInfo(r.Form[cavName][i]) |
| case "PeerBlessingsCaveat": |
| caveat, err = newPeerBlessingsCaveatInfo(r.Form[cavName][i]) |
| case "none": |
| continue |
| default: |
| err = errors.New("caveat does not exist") |
| } |
| if err != nil { |
| return nil, fmt.Errorf("unable to create caveat %s: %v", cavName, err) |
| } |
| caveats = append(caveats, caveat) |
| } |
| return caveats, nil |
| } |
| |
| func newExpiryCaveatInfo(timestamp, utcOffset string) (CaveatInfo, error) { |
| var empty CaveatInfo |
| t, err := time.Parse("2006-01-02T15:04", timestamp) |
| if err != nil { |
| return empty, fmt.Errorf("parseTime failed: %v", err) |
| } |
| // utcOffset is returned as minutes from JS, so we need to parse it to a duration. |
| offset, err := time.ParseDuration(utcOffset + "m") |
| if err != nil { |
| return empty, fmt.Errorf("failed to parse duration: %v", err) |
| } |
| return CaveatInfo{"Expiry", []interface{}{t.Add(offset)}}, nil |
| } |
| |
| func newMethodCaveatInfo(methodsCSV string) (CaveatInfo, error) { |
| methods := strings.Split(methodsCSV, ",") |
| if len(methods) < 1 { |
| return CaveatInfo{}, fmt.Errorf("must pass at least one method") |
| } |
| var ifaces []interface{} |
| for _, m := range methods { |
| ifaces = append(ifaces, m) |
| } |
| return CaveatInfo{"Method", ifaces}, nil |
| } |
| |
| func newPeerBlessingsCaveatInfo(patternsCSV string) (CaveatInfo, error) { |
| patterns := strings.Split(patternsCSV, ",") |
| if len(patterns) < 1 { |
| return CaveatInfo{}, fmt.Errorf("must pass at least one peer blessing pattern") |
| } |
| var ifaces []interface{} |
| for _, p := range patterns { |
| ifaces = append(ifaces, security.BlessingPattern(p)) |
| } |
| return CaveatInfo{"PeerBlessings", ifaces}, nil |
| } |
| |
| func newRevocationCaveatInfo() CaveatInfo { |
| return CaveatInfo{Type: "Revocation"} |
| } |