2020-02-15 22:13:50 +00:00
|
|
|
package dehub
|
|
|
|
|
|
|
|
import (
|
2020-04-11 23:10:18 +00:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
|
2020-03-29 21:16:36 +00:00
|
|
|
"dehub.dev/src/dehub.git/accessctl"
|
|
|
|
"dehub.dev/src/dehub.git/fs"
|
|
|
|
"dehub.dev/src/dehub.git/sigcred"
|
2020-02-15 22:13:50 +00:00
|
|
|
|
|
|
|
yaml "gopkg.in/yaml.v2"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Account represents a single account defined in the Config.
|
|
|
|
type Account struct {
|
|
|
|
ID string `yaml:"id"`
|
|
|
|
Signifiers []sigcred.Signifier `yaml:"signifiers"`
|
|
|
|
Meta map[string]string `yaml:"meta,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Config represents the structure of the main dehub configuration file, and is
|
|
|
|
// used to marshal/unmarshal the yaml file.
|
|
|
|
type Config struct {
|
2020-03-18 22:35:32 +00:00
|
|
|
Accounts []Account `yaml:"accounts"`
|
|
|
|
AccessControls []accessctl.AccessControl `yaml:"access_controls"`
|
2020-02-15 22:13:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *Repo) loadConfig(fs fs.FS) (Config, error) {
|
|
|
|
rc, err := fs.Open(ConfigPath)
|
|
|
|
if err != nil {
|
|
|
|
return Config{}, fmt.Errorf("could not open config.yml: %w", err)
|
|
|
|
}
|
|
|
|
defer rc.Close()
|
|
|
|
|
|
|
|
var cfg Config
|
|
|
|
if err := yaml.NewDecoder(rc).Decode(&cfg); err != nil {
|
|
|
|
return cfg, fmt.Errorf("could not decode config.yml: %w", err)
|
|
|
|
}
|
|
|
|
|
2020-03-18 22:35:32 +00:00
|
|
|
// older config versions also had access_controls be an array, but not using
|
|
|
|
// the action field. So filter out array elements without the action field.
|
|
|
|
acl := cfg.AccessControls
|
|
|
|
cfg.AccessControls = cfg.AccessControls[:0]
|
|
|
|
for _, ac := range acl {
|
|
|
|
if ac.Action == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
cfg.AccessControls = append(cfg.AccessControls, ac)
|
|
|
|
}
|
|
|
|
|
2020-02-15 22:13:50 +00:00
|
|
|
// TODO validate Config
|
|
|
|
|
|
|
|
return cfg, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// LoadConfig loads the Config object from the HEAD of the repo, or directly
|
|
|
|
// from the filesystem if there is no HEAD yet.
|
|
|
|
func (r *Repo) LoadConfig() (Config, error) {
|
normalize how git commits are interacted with, including changing VerifyComit -> VerifyCommits
---
type: change
message: |-
normalize how git commits are interacted with, including changing VerifyComit -> VerifyCommits
This commit attempts to normalize git commit interactions in order to reduce
the amount of manual `GitRepo.CommitObject`, `GitRepo.TreeObject`,
`Commit.UnmarshalText`, and `Commit.Interface` calls are done, by creating a
single structure (`GitCommit`) which holds the output of those calls, and is
only created by a single method (`GetGitCommit`), which is then used by a bunch
of other methods to expand its functionality, including implementing a range
request which can be used by verify and the pre-receive hook (though it's only
used by the hook, currently).
change_hash: AMae4PL6+jrxhn2KEGHejstcdT37Gw/jjkl/UuovHcgd
credentials:
- type: pgp_signature
pub_key_id: 95C46FA6A41148AC
body: iQIzBAABAgAdFiEEJ6tQKp6olvZKJ0lwlcRvpqQRSKwFAl5uhvoACgkQlcRvpqQRSKzJrhAAqi2LEQVTyVktfsOBv/CZmefclLLqWTChVoeIZt2EAGDDGygmrx88hI0SEAviOzPMn0kiZFDeY5k7ICJMhJ9RVDU9WjH7fbOboMJW19rVhx6Ke/M2ERtrT0OFLRmFVJVDM0P8SEheQvR3HE/iiypBICVCtp+meHEq9mOJWZlZnoCqMaulAy/Nnq4N1VD0yPPlr16+yxMqedKHcgKbcH8K61ltNAjXDT+tCWwCq1huA5MVSuTm5EwqIeKPN6JKgwATv8Ku2GhYZWHSGUwecP1J3x2XTDPeChCQVDpC232Pxwk8z/D36F3J/XOfkdl0QYQ077xL1IJfYOnuuHir47CokDf3G0XCQnJ/+X4pZdtP387rc045o/2bhUi2U4eJ5HgS7Hvyi6EApT0Czv7SeJePTvdnRUYse8ZYuIwYXj5GWWxnbKQzLpyjcHdQc2a3B3RN84zXqqAOS6ObFrFPZQIfz2rfQojZN8kvcmUvYhJXSaT65XmqFjyJ4n6grrEnK/N+MfbnpzyF/yvlzxWPqGFQOQj9meosbTAdgZbmdwYqa5r1ee8DmlkzNJJxze96h503a733yciN8Ef4hGZNlRV6YFegkK/cCgKaA4NCEALKb1t0Uri5gnPldXk4HsPF+23GANbE7mjytY8ra3fhXG4VhaFt/WsLg3Bu7djQ0H74y+g=
account: mediocregopher
2020-03-15 19:50:24 +00:00
|
|
|
headFS, err := r.headFS()
|
2020-02-15 22:13:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return Config{}, fmt.Errorf("error retrieving repo HEAD: %w", err)
|
|
|
|
}
|
|
|
|
return r.loadConfig(headFS)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *Repo) signifierForCredential(fs fs.FS, cred sigcred.Credential) (sigcred.SignifierInterface, error) {
|
|
|
|
cfg, err := r.loadConfig(fs)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error loading config: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var account Account
|
|
|
|
var ok bool
|
|
|
|
for _, account = range cfg.Accounts {
|
|
|
|
if account.ID == cred.AccountID {
|
|
|
|
ok = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("no account object for account id %q present in config", cred.AccountID)
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, sig := range account.Signifiers {
|
2020-03-13 21:24:46 +00:00
|
|
|
if sigInt, err := sig.Interface(cred.AccountID); err != nil {
|
2020-02-15 22:13:50 +00:00
|
|
|
return nil, fmt.Errorf("error converting signifier index:%d to inteface: %w", i, err)
|
|
|
|
} else if ok, err := sigInt.Signed(fs, cred); err != nil {
|
|
|
|
return nil, fmt.Errorf("error checking if signfier index:%d signed credential: %w", i, err)
|
|
|
|
} else if ok {
|
|
|
|
return sigInt, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, errors.New("no signifier found for credential")
|
|
|
|
}
|