git-bug/repository/mock_repo.go

471 lines
9.8 KiB
Go
Raw Normal View History

2018-07-12 10:55:13 +03:00
package repository
import (
2021-01-04 01:59:25 +03:00
"bytes"
"crypto/sha1"
"fmt"
2020-06-27 23:00:15 +03:00
"strings"
2020-12-08 15:15:21 +03:00
"sync"
"github.com/99designs/keyring"
2020-12-08 16:42:13 +03:00
"github.com/blevesearch/bleve"
2020-12-01 23:19:23 +03:00
"github.com/go-git/go-billy/v5"
"github.com/go-git/go-billy/v5/memfs"
2021-01-04 01:59:25 +03:00
"golang.org/x/crypto/openpgp"
"github.com/MichaelMure/git-bug/util/lamport"
2018-07-12 10:55:13 +03:00
)
var _ ClockedRepo = &mockRepo{}
var _ TestedRepo = &mockRepo{}
2019-02-19 02:19:27 +03:00
// mockRepo defines an instance of Repo that can be used for testing.
type mockRepo struct {
*mockRepoConfig
*mockRepoKeyring
*mockRepoCommon
2020-12-01 23:19:23 +03:00
*mockRepoStorage
2020-12-08 16:42:13 +03:00
*mockRepoBleve
*mockRepoData
*mockRepoClock
*mockRepoTest
}
2018-07-12 10:55:13 +03:00
func (m *mockRepo) Close() error { return nil }
2020-12-08 16:42:13 +03:00
func NewMockRepo() *mockRepo {
return &mockRepo{
mockRepoConfig: NewMockRepoConfig(),
mockRepoKeyring: NewMockRepoKeyring(),
mockRepoCommon: NewMockRepoCommon(),
2020-12-01 23:19:23 +03:00
mockRepoStorage: NewMockRepoStorage(),
2020-12-08 16:42:13 +03:00
mockRepoBleve: newMockRepoBleve(),
mockRepoData: NewMockRepoData(),
mockRepoClock: NewMockRepoClock(),
mockRepoTest: NewMockRepoTest(),
}
}
var _ RepoConfig = &mockRepoConfig{}
type mockRepoConfig struct {
localConfig *MemConfig
globalConfig *MemConfig
}
func NewMockRepoConfig() *mockRepoConfig {
return &mockRepoConfig{
localConfig: NewMemConfig(),
globalConfig: NewMemConfig(),
}
}
// LocalConfig give access to the repository scoped configuration
func (r *mockRepoConfig) LocalConfig() Config {
return r.localConfig
}
// GlobalConfig give access to the git global configuration
func (r *mockRepoConfig) GlobalConfig() Config {
return r.globalConfig
}
// AnyConfig give access to a merged local/global configuration
func (r *mockRepoConfig) AnyConfig() ConfigRead {
return mergeConfig(r.localConfig, r.globalConfig)
}
var _ RepoKeyring = &mockRepoKeyring{}
type mockRepoKeyring struct {
keyring *keyring.ArrayKeyring
}
func NewMockRepoKeyring() *mockRepoKeyring {
return &mockRepoKeyring{
keyring: keyring.NewArrayKeyring(nil),
}
}
// Keyring give access to a user-wide storage for secrets
func (r *mockRepoKeyring) Keyring() Keyring {
return r.keyring
}
var _ RepoCommon = &mockRepoCommon{}
type mockRepoCommon struct{}
func NewMockRepoCommon() *mockRepoCommon {
return &mockRepoCommon{}
}
func (r *mockRepoCommon) GetUserName() (string, error) {
return "René Descartes", nil
2018-07-12 10:55:13 +03:00
}
// GetUserEmail returns the email address that the user has used to configure git.
func (r *mockRepoCommon) GetUserEmail() (string, error) {
return "user@example.com", nil
}
2018-07-12 10:55:13 +03:00
// GetCoreEditor returns the name of the editor that the user has used to configure git.
func (r *mockRepoCommon) GetCoreEditor() (string, error) {
return "vi", nil
}
2018-07-12 10:55:13 +03:00
2019-05-25 16:13:40 +03:00
// GetRemotes returns the configured remotes repositories.
func (r *mockRepoCommon) GetRemotes() (map[string]string, error) {
2019-05-25 16:13:40 +03:00
return map[string]string{
"origin": "git://github.com/MichaelMure/git-bug",
}, nil
}
2020-12-01 23:19:23 +03:00
var _ RepoStorage = &mockRepoStorage{}
type mockRepoStorage struct {
localFs billy.Filesystem
}
func NewMockRepoStorage() *mockRepoStorage {
return &mockRepoStorage{localFs: memfs.New()}
}
func (m *mockRepoStorage) LocalStorage() billy.Filesystem {
return m.localFs
}
2020-12-08 16:42:13 +03:00
var _ RepoBleve = &mockRepoBleve{}
type mockRepoBleve struct {
indexesMutex sync.Mutex
indexes map[string]bleve.Index
}
func newMockRepoBleve() *mockRepoBleve {
return &mockRepoBleve{
indexes: make(map[string]bleve.Index),
}
}
func (m *mockRepoBleve) GetBleveIndex(name string) (bleve.Index, error) {
m.indexesMutex.Lock()
defer m.indexesMutex.Unlock()
if index, ok := m.indexes[name]; ok {
return index, nil
}
mapping := bleve.NewIndexMapping()
mapping.DefaultAnalyzer = "en"
index, err := bleve.NewMemOnly(mapping)
if err != nil {
return nil, err
}
m.indexes[name] = index
return index, nil
}
func (m *mockRepoBleve) ClearBleveIndex(name string) error {
m.indexesMutex.Lock()
defer m.indexesMutex.Unlock()
delete(m.indexes, name)
return nil
}
var _ RepoData = &mockRepoData{}
type commit struct {
treeHash Hash
parents []Hash
2021-01-04 01:59:25 +03:00
sig string
}
type mockRepoData struct {
blobs map[Hash][]byte
trees map[Hash]string
commits map[Hash]commit
refs map[string]Hash
}
func NewMockRepoData() *mockRepoData {
return &mockRepoData{
blobs: make(map[Hash][]byte),
trees: make(map[Hash]string),
commits: make(map[Hash]commit),
refs: make(map[string]Hash),
}
}
2021-02-05 13:18:38 +03:00
func (r *mockRepoData) FetchRefs(remote string, prefix string) (string, error) {
2021-01-25 14:39:34 +03:00
panic("implement me")
}
2021-01-04 01:59:25 +03:00
// PushRefs push git refs to a remote
2021-02-05 13:18:38 +03:00
func (r *mockRepoData) PushRefs(remote string, prefix string) (string, error) {
2021-01-25 14:39:34 +03:00
panic("implement me")
}
func (r *mockRepoData) StoreData(data []byte) (Hash, error) {
rawHash := sha1.Sum(data)
hash := Hash(fmt.Sprintf("%x", rawHash))
r.blobs[hash] = data
return hash, nil
}
func (r *mockRepoData) ReadData(hash Hash) ([]byte, error) {
data, ok := r.blobs[hash]
if !ok {
2018-07-19 14:33:16 +03:00
return nil, fmt.Errorf("unknown hash")
}
return data, nil
}
func (r *mockRepoData) StoreTree(entries []TreeEntry) (Hash, error) {
buffer := prepareTreeEntries(entries)
rawHash := sha1.Sum(buffer.Bytes())
hash := Hash(fmt.Sprintf("%x", rawHash))
r.trees[hash] = buffer.String()
return hash, nil
}
2021-01-04 01:59:25 +03:00
func (r *mockRepoData) ReadTree(hash Hash) ([]TreeEntry, error) {
var data string
data, ok := r.trees[hash]
if !ok {
// Git will understand a commit hash to reach a tree
commit, ok := r.commits[hash]
if !ok {
return nil, fmt.Errorf("unknown hash")
}
data, ok = r.trees[commit.treeHash]
if !ok {
return nil, fmt.Errorf("unknown hash")
}
}
2021-01-04 01:59:25 +03:00
return readTreeEntries(data)
}
func (r *mockRepoData) StoreCommit(treeHash Hash, parents ...Hash) (Hash, error) {
return r.StoreSignedCommit(treeHash, nil, parents...)
}
2021-01-04 01:59:25 +03:00
func (r *mockRepoData) StoreSignedCommit(treeHash Hash, signKey *openpgp.Entity, parents ...Hash) (Hash, error) {
hasher := sha1.New()
hasher.Write([]byte(treeHash))
for _, parent := range parents {
hasher.Write([]byte(parent))
}
rawHash := hasher.Sum(nil)
hash := Hash(fmt.Sprintf("%x", rawHash))
2021-01-04 01:59:25 +03:00
c := commit{
treeHash: treeHash,
2021-01-04 01:59:25 +03:00
parents: parents,
}
if signKey != nil {
// unlike go-git, we only sign the tree hash for simplicity instead of all the fields (parents ...)
var sig bytes.Buffer
if err := openpgp.DetachSign(&sig, signKey, strings.NewReader(string(treeHash)), nil); err != nil {
return "", err
}
c.sig = sig.String()
}
2021-01-04 01:59:25 +03:00
r.commits[hash] = c
return hash, nil
}
2021-01-04 01:59:25 +03:00
func (r *mockRepoData) ReadCommit(hash Hash) (Commit, error) {
c, ok := r.commits[hash]
if !ok {
return Commit{}, fmt.Errorf("unknown commit")
}
result := Commit{
Hash: hash,
Parents: c.parents,
TreeHash: c.treeHash,
}
if c.sig != "" {
2021-02-09 12:46:33 +03:00
// Note: this is actually incorrect as the signed data should be the full commit (+comment, +date ...)
// but only the tree hash work for our purpose here.
2021-01-04 01:59:25 +03:00
result.SignedData = strings.NewReader(string(c.treeHash))
result.Signature = strings.NewReader(c.sig)
}
return result, nil
}
func (r *mockRepoData) GetTreeHash(commit Hash) (Hash, error) {
c, ok := r.commits[commit]
if !ok {
return "", fmt.Errorf("unknown commit")
}
return c.treeHash, nil
}
func (r *mockRepoData) ResolveRef(ref string) (Hash, error) {
h, ok := r.refs[ref]
if !ok {
return "", fmt.Errorf("unknown ref")
}
return h, nil
}
func (r *mockRepoData) UpdateRef(ref string, hash Hash) error {
r.refs[ref] = hash
return nil
}
func (r *mockRepoData) RemoveRef(ref string) error {
delete(r.refs, ref)
return nil
}
func (r *mockRepoData) ListRefs(refPrefix string) ([]string, error) {
2020-06-27 23:00:15 +03:00
var keys []string
for k := range r.refs {
2020-09-08 15:31:40 +03:00
if strings.HasPrefix(k, refPrefix) {
2020-06-27 23:00:15 +03:00
keys = append(keys, k)
}
}
return keys, nil
}
2021-01-04 01:59:25 +03:00
func (r *mockRepoData) RefExist(ref string) (bool, error) {
_, exist := r.refs[ref]
return exist, nil
}
2021-01-04 01:59:25 +03:00
func (r *mockRepoData) CopyRef(source string, dest string) error {
hash, exist := r.refs[source]
2021-01-04 01:59:25 +03:00
if !exist {
return fmt.Errorf("Unknown ref")
}
2021-01-04 01:59:25 +03:00
r.refs[dest] = hash
return nil
}
2018-07-17 02:52:56 +03:00
func (r *mockRepoData) FindCommonAncestor(hash1 Hash, hash2 Hash) (Hash, error) {
ancestor1 := []Hash{hash1}
2020-06-27 23:00:15 +03:00
for hash1 != "" {
c, ok := r.commits[hash1]
if !ok {
return "", fmt.Errorf("unknown commit %v", hash1)
}
if len(c.parents) == 0 {
break
}
ancestor1 = append(ancestor1, c.parents[0])
hash1 = c.parents[0]
2020-06-27 23:00:15 +03:00
}
for {
for _, ancestor := range ancestor1 {
if ancestor == hash2 {
return ancestor, nil
}
}
c, ok := r.commits[hash2]
if !ok {
return "", fmt.Errorf("unknown commit %v", hash1)
}
if c.parents[0] == "" {
2020-06-27 23:00:15 +03:00
return "", fmt.Errorf("no ancestor found")
}
hash2 = c.parents[0]
2020-06-27 23:00:15 +03:00
}
2018-07-17 02:52:56 +03:00
}
2021-01-04 01:59:25 +03:00
func (r *mockRepoData) ListCommits(ref string) ([]Hash, error) {
return nonNativeListCommits(r, ref)
2018-07-17 02:52:56 +03:00
}
var _ RepoClock = &mockRepoClock{}
2020-12-01 23:19:23 +03:00
type mockRepoClock struct {
2020-12-08 15:15:21 +03:00
mu sync.Mutex
clocks map[string]lamport.Clock
}
func NewMockRepoClock() *mockRepoClock {
return &mockRepoClock{
clocks: make(map[string]lamport.Clock),
}
}
func (r *mockRepoClock) AllClocks() (map[string]lamport.Clock, error) {
return r.clocks, nil
}
func (r *mockRepoClock) GetOrCreateClock(name string) (lamport.Clock, error) {
2020-12-08 15:15:21 +03:00
r.mu.Lock()
defer r.mu.Unlock()
if c, ok := r.clocks[name]; ok {
return c, nil
}
c := lamport.NewMemClock()
r.clocks[name] = c
return c, nil
}
func (r *mockRepoClock) Increment(name string) (lamport.Time, error) {
c, err := r.GetOrCreateClock(name)
if err != nil {
return lamport.Time(0), err
}
return c.Increment()
}
func (r *mockRepoClock) Witness(name string, time lamport.Time) error {
c, err := r.GetOrCreateClock(name)
if err != nil {
return err
}
return c.Witness(time)
}
var _ repoTest = &mockRepoTest{}
type mockRepoTest struct{}
func NewMockRepoTest() *mockRepoTest {
return &mockRepoTest{}
}
func (r *mockRepoTest) AddRemote(name string, url string) error {
panic("implement me")
}
func (r mockRepoTest) GetLocalRemote() string {
panic("implement me")
}
func (r mockRepoTest) EraseFromDisk() error {
// nothing to do
return nil
}