2019-09-16 23:16:33 +03:00
|
|
|
package core
|
|
|
|
|
|
|
|
import (
|
2019-10-24 21:16:51 +03:00
|
|
|
"crypto/sha256"
|
2019-11-10 16:46:55 +03:00
|
|
|
"errors"
|
2019-09-16 23:16:33 +03:00
|
|
|
"fmt"
|
|
|
|
"regexp"
|
2019-11-10 16:46:55 +03:00
|
|
|
"sort"
|
2019-09-16 23:16:33 +03:00
|
|
|
"strings"
|
2019-11-08 16:49:24 +03:00
|
|
|
"time"
|
|
|
|
|
2019-11-08 16:55:27 +03:00
|
|
|
"github.com/MichaelMure/git-bug/entity"
|
2019-09-17 14:59:40 +03:00
|
|
|
"github.com/MichaelMure/git-bug/repository"
|
2019-09-16 23:16:33 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
tokenConfigKeyPrefix = "git-bug.token"
|
2019-10-24 21:16:51 +03:00
|
|
|
tokenValueKey = "value"
|
|
|
|
tokenTargetKey = "target"
|
2019-11-08 16:49:24 +03:00
|
|
|
tokenCreateTimeKey = "createtime"
|
2019-09-16 23:16:33 +03:00
|
|
|
)
|
|
|
|
|
2019-11-10 16:46:55 +03:00
|
|
|
var ErrTokenNotExist = errors.New("token doesn't exist")
|
|
|
|
|
|
|
|
func NewErrMultipleMatchToken(matching []entity.Id) *entity.ErrMultipleMatch {
|
|
|
|
return entity.NewErrMultipleMatch("token", matching)
|
|
|
|
}
|
|
|
|
|
2019-10-12 12:10:44 +03:00
|
|
|
// Token holds an API access token data
|
2019-09-16 23:16:33 +03:00
|
|
|
type Token struct {
|
2019-11-08 16:49:24 +03:00
|
|
|
Value string
|
|
|
|
Target string
|
|
|
|
CreateTime time.Time
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
|
|
|
|
2019-09-17 14:59:40 +03:00
|
|
|
// NewToken instantiate a new token
|
2019-11-08 16:49:24 +03:00
|
|
|
func NewToken(value, target string) *Token {
|
2019-11-10 16:46:55 +03:00
|
|
|
return &Token{
|
2019-11-08 16:49:24 +03:00
|
|
|
Value: value,
|
|
|
|
Target: target,
|
|
|
|
CreateTime: time.Now(),
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
2019-10-24 21:16:51 +03:00
|
|
|
}
|
|
|
|
|
2019-11-10 16:46:55 +03:00
|
|
|
func (t *Token) ID() entity.Id {
|
2019-11-19 23:23:03 +03:00
|
|
|
sum := sha256.Sum256([]byte(t.Target + t.Value))
|
2019-11-10 16:46:55 +03:00
|
|
|
return entity.Id(fmt.Sprintf("%x", sum))
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
|
|
|
|
2019-09-17 14:59:40 +03:00
|
|
|
// Validate ensure token important fields are valid
|
|
|
|
func (t *Token) Validate() error {
|
2019-09-16 23:16:33 +03:00
|
|
|
if t.Value == "" {
|
2019-10-12 12:10:44 +03:00
|
|
|
return fmt.Errorf("missing value")
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
|
|
|
if t.Target == "" {
|
2019-10-12 12:10:44 +03:00
|
|
|
return fmt.Errorf("missing target")
|
|
|
|
}
|
2019-11-10 17:50:56 +03:00
|
|
|
if t.CreateTime.IsZero() || t.CreateTime.Equal(time.Time{}) {
|
2019-11-08 16:49:24 +03:00
|
|
|
return fmt.Errorf("missing creation time")
|
|
|
|
}
|
2019-11-10 16:46:55 +03:00
|
|
|
if !TargetExist(t.Target) {
|
2019-10-12 12:10:44 +03:00
|
|
|
return fmt.Errorf("unknown target")
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-10 16:46:55 +03:00
|
|
|
// LoadToken loads a token from the repo config
|
|
|
|
func LoadToken(repo repository.RepoCommon, id entity.Id) (*Token, error) {
|
2019-10-24 21:16:51 +03:00
|
|
|
keyPrefix := fmt.Sprintf("git-bug.token.%s.", id)
|
2019-09-17 14:59:40 +03:00
|
|
|
|
2019-10-12 12:10:44 +03:00
|
|
|
// read token config pairs
|
2019-11-10 16:46:55 +03:00
|
|
|
rawconfigs, err := repo.GlobalConfig().ReadAll(keyPrefix)
|
2019-10-12 12:10:44 +03:00
|
|
|
if err != nil {
|
2019-11-10 16:46:55 +03:00
|
|
|
// Not exactly right due to the limitation of ReadAll()
|
|
|
|
return nil, ErrTokenNotExist
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
|
|
|
|
2019-09-17 14:59:40 +03:00
|
|
|
// trim key prefix
|
2019-11-10 16:46:55 +03:00
|
|
|
configs := make(map[string]string)
|
|
|
|
for key, value := range rawconfigs {
|
2019-10-12 12:10:44 +03:00
|
|
|
newKey := strings.TrimPrefix(key, keyPrefix)
|
|
|
|
configs[newKey] = value
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
|
|
|
|
2019-11-10 16:46:55 +03:00
|
|
|
token := &Token{}
|
|
|
|
|
|
|
|
token.Value = configs[tokenValueKey]
|
|
|
|
token.Target = configs[tokenTargetKey]
|
|
|
|
if createTime, ok := configs[tokenCreateTimeKey]; ok {
|
|
|
|
if t, err := repository.ParseTimestamp(createTime); err == nil {
|
|
|
|
token.CreateTime = t
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return token, nil
|
|
|
|
}
|
2019-10-24 21:16:51 +03:00
|
|
|
|
2019-11-10 16:46:55 +03:00
|
|
|
// LoadTokenPrefix load a token from the repo config with a prefix
|
|
|
|
func LoadTokenPrefix(repo repository.RepoCommon, prefix string) (*Token, error) {
|
|
|
|
tokens, err := ListTokens(repo)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2019-10-24 21:16:51 +03:00
|
|
|
}
|
|
|
|
|
2019-11-10 16:46:55 +03:00
|
|
|
// preallocate but empty
|
|
|
|
matching := make([]entity.Id, 0, 5)
|
|
|
|
|
|
|
|
for _, id := range tokens {
|
|
|
|
if id.HasPrefix(prefix) {
|
|
|
|
matching = append(matching, id)
|
|
|
|
}
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
|
|
|
|
2019-11-10 16:46:55 +03:00
|
|
|
if len(matching) > 1 {
|
|
|
|
return nil, NewErrMultipleMatchToken(matching)
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
|
|
|
|
2019-11-10 16:46:55 +03:00
|
|
|
if len(matching) == 0 {
|
|
|
|
return nil, ErrTokenNotExist
|
2019-11-08 16:49:24 +03:00
|
|
|
}
|
2019-11-10 16:46:55 +03:00
|
|
|
|
|
|
|
return LoadToken(repo, matching[0])
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
|
|
|
|
2019-11-23 21:20:19 +03:00
|
|
|
// ListTokens list all existing token ids
|
2019-11-10 16:46:55 +03:00
|
|
|
func ListTokens(repo repository.RepoCommon) ([]entity.Id, error) {
|
2019-11-08 16:49:24 +03:00
|
|
|
configs, err := repo.GlobalConfig().ReadAll(tokenConfigKeyPrefix + ".")
|
2019-10-12 12:10:44 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
re, err := regexp.Compile(tokenConfigKeyPrefix + `.([^.]+)`)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
set := make(map[string]interface{})
|
|
|
|
|
|
|
|
for key := range configs {
|
|
|
|
res := re.FindStringSubmatch(key)
|
|
|
|
|
|
|
|
if res == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
set[res[1]] = nil
|
|
|
|
}
|
|
|
|
|
2019-11-10 16:46:55 +03:00
|
|
|
result := make([]entity.Id, 0, len(set))
|
2019-09-16 23:16:33 +03:00
|
|
|
for key := range set {
|
2019-11-10 16:46:55 +03:00
|
|
|
result = append(result, entity.Id(key))
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
|
|
|
|
2019-11-10 16:46:55 +03:00
|
|
|
sort.Sort(entity.Alphabetical(result))
|
|
|
|
|
2019-09-16 23:16:33 +03:00
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
2019-11-23 21:20:19 +03:00
|
|
|
// ListTokensWithTarget list all token ids associated with the target
|
|
|
|
func ListTokensWithTarget(repo repository.RepoCommon, target string) ([]entity.Id, error) {
|
|
|
|
var ids []entity.Id
|
|
|
|
tokensIds, err := ListTokens(repo)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tokenId := range tokensIds {
|
|
|
|
token, err := LoadToken(repo, tokenId)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if token.Target == target {
|
|
|
|
ids = append(ids, tokenId)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ids, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// LoadTokens load all existing tokens
|
|
|
|
func LoadTokens(repo repository.RepoCommon) ([]*Token, error) {
|
|
|
|
tokensIds, err := ListTokens(repo)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var tokens []*Token
|
|
|
|
for _, id := range tokensIds {
|
|
|
|
token, err := LoadToken(repo, id)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
tokens = append(tokens, token)
|
|
|
|
}
|
|
|
|
return tokens, nil
|
|
|
|
}
|
|
|
|
|
2019-11-24 16:38:44 +03:00
|
|
|
// LoadTokensWithTarget load all existing tokens for a given target
|
|
|
|
func LoadTokensWithTarget(repo repository.RepoCommon, target string) ([]*Token, error) {
|
|
|
|
tokensIds, err := ListTokens(repo)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var tokens []*Token
|
|
|
|
for _, id := range tokensIds {
|
|
|
|
token, err := LoadToken(repo, id)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if token.Target == target {
|
|
|
|
tokens = append(tokens, token)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return tokens, nil
|
|
|
|
}
|
|
|
|
|
2019-11-23 21:20:19 +03:00
|
|
|
// TokenIdExist return wether token id exist or not
|
|
|
|
func TokenIdExist(repo repository.RepoCommon, id entity.Id) bool {
|
|
|
|
_, err := LoadToken(repo, id)
|
|
|
|
return err == nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// TokenExist return wether there is a token with a certain value or not
|
|
|
|
func TokenExist(repo repository.RepoCommon, value string) bool {
|
|
|
|
tokens, err := LoadTokens(repo)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
for _, token := range tokens {
|
|
|
|
if token.Value == value {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// TokenExistWithTarget same as TokenExist but restrict search for a given target
|
|
|
|
func TokenExistWithTarget(repo repository.RepoCommon, value string, target string) bool {
|
2019-11-24 16:39:02 +03:00
|
|
|
tokens, err := LoadTokensWithTarget(repo, target)
|
2019-11-23 21:20:19 +03:00
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
for _, token := range tokens {
|
2019-11-24 16:39:02 +03:00
|
|
|
if token.Value == value {
|
2019-11-23 21:20:19 +03:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2019-11-08 16:49:24 +03:00
|
|
|
// StoreToken stores a token in the repo config
|
|
|
|
func StoreToken(repo repository.RepoCommon, token *Token) error {
|
2019-11-10 16:46:55 +03:00
|
|
|
storeValueKey := fmt.Sprintf("git-bug.token.%s.%s", token.ID().String(), tokenValueKey)
|
2019-11-08 16:49:24 +03:00
|
|
|
err := repo.GlobalConfig().StoreString(storeValueKey, token.Value)
|
2019-10-24 21:16:51 +03:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-11-10 16:46:55 +03:00
|
|
|
storeTargetKey := fmt.Sprintf("git-bug.token.%s.%s", token.ID().String(), tokenTargetKey)
|
2019-11-08 16:49:24 +03:00
|
|
|
err = repo.GlobalConfig().StoreString(storeTargetKey, token.Target)
|
2019-09-16 23:16:33 +03:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-11-10 16:46:55 +03:00
|
|
|
createTimeKey := fmt.Sprintf("git-bug.token.%s.%s", token.ID().String(), tokenCreateTimeKey)
|
|
|
|
return repo.GlobalConfig().StoreTimestamp(createTimeKey, token.CreateTime)
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
|
|
|
|
2019-09-17 14:59:40 +03:00
|
|
|
// RemoveToken removes a token from the repo config
|
2019-11-10 16:46:55 +03:00
|
|
|
func RemoveToken(repo repository.RepoCommon, id entity.Id) error {
|
2019-10-24 21:16:51 +03:00
|
|
|
keyPrefix := fmt.Sprintf("git-bug.token.%s", id)
|
2019-11-08 16:49:24 +03:00
|
|
|
return repo.GlobalConfig().RemoveAll(keyPrefix)
|
2019-09-16 23:16:33 +03:00
|
|
|
}
|
2019-11-24 16:38:44 +03:00
|
|
|
|
|
|
|
// LoadOrCreateToken will try to load a token matching the same value or create it
|
|
|
|
func LoadOrCreateToken(repo repository.RepoCommon, target, tokenValue string) (*Token, error) {
|
|
|
|
tokens, err := LoadTokensWithTarget(repo, target)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, token := range tokens {
|
|
|
|
if token.Value == tokenValue {
|
|
|
|
return token, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
token := NewToken(tokenValue, target)
|
|
|
|
err = StoreToken(repo, token)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return token, nil
|
|
|
|
}
|