2018-07-12 13:44:46 +03:00
|
|
|
package bug
|
|
|
|
|
2018-07-13 17:13:40 +03:00
|
|
|
import (
|
2018-07-13 22:21:24 +03:00
|
|
|
"fmt"
|
|
|
|
"github.com/MichaelMure/git-bug/repository"
|
|
|
|
"github.com/MichaelMure/git-bug/util"
|
2018-07-13 17:13:40 +03:00
|
|
|
"github.com/kevinburke/go.uuid"
|
|
|
|
)
|
|
|
|
|
2018-07-13 22:21:24 +03:00
|
|
|
const BugsRefPattern = "refs/bugs/"
|
|
|
|
|
2018-07-13 17:13:40 +03:00
|
|
|
// Bug hold the data of a bug thread, organized in a way close to
|
|
|
|
// how it will be persisted inside Git. This is the datastructure
|
|
|
|
// used for merge of two different version.
|
|
|
|
type Bug struct {
|
|
|
|
// Id used as unique identifier
|
2018-07-13 22:21:24 +03:00
|
|
|
id uuid.UUID
|
|
|
|
|
|
|
|
lastCommit util.Hash
|
2018-07-13 17:13:40 +03:00
|
|
|
|
2018-07-13 22:21:24 +03:00
|
|
|
// TODO: need a way to order bugs, probably a Lamport clock
|
2018-07-13 17:13:40 +03:00
|
|
|
|
2018-07-13 22:21:24 +03:00
|
|
|
packs []OperationPack
|
2018-07-13 17:13:40 +03:00
|
|
|
|
2018-07-13 22:21:24 +03:00
|
|
|
staging OperationPack
|
2018-07-12 13:44:46 +03:00
|
|
|
}
|
2018-07-12 16:14:37 +03:00
|
|
|
|
2018-07-13 17:13:40 +03:00
|
|
|
// Create a new Bug
|
|
|
|
func NewBug() (*Bug, error) {
|
|
|
|
// Creating UUID Version 4
|
|
|
|
id, err := uuid.ID4()
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Bug{
|
2018-07-13 22:21:24 +03:00
|
|
|
id: id,
|
|
|
|
lastCommit: "",
|
2018-07-13 17:13:40 +03:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsValid check if the Bug data is valid
|
|
|
|
func (bug *Bug) IsValid() bool {
|
|
|
|
// non-empty
|
2018-07-13 22:21:24 +03:00
|
|
|
if len(bug.packs) == 0 && bug.staging.IsEmpty() {
|
2018-07-13 17:13:40 +03:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if each pack is valid
|
2018-07-13 22:21:24 +03:00
|
|
|
for _, pack := range bug.packs {
|
2018-07-13 17:13:40 +03:00
|
|
|
if !pack.IsValid() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-13 22:21:24 +03:00
|
|
|
// check if staging is valid if needed
|
|
|
|
if !bug.staging.IsEmpty() {
|
|
|
|
if !bug.staging.IsValid() {
|
2018-07-13 17:48:55 +03:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-13 17:13:40 +03:00
|
|
|
// The very first Op should be a CREATE
|
2018-07-13 17:48:55 +03:00
|
|
|
firstOp := bug.firstOp()
|
|
|
|
if firstOp == nil || firstOp.OpType() != CREATE {
|
2018-07-13 17:13:40 +03:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that there is no more CREATE op
|
|
|
|
it := NewOperationIterator(bug)
|
|
|
|
createCount := 0
|
|
|
|
for it.Next() {
|
|
|
|
if it.Value().OpType() == CREATE {
|
|
|
|
createCount++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if createCount != 1 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bug *Bug) Append(op Operation) {
|
2018-07-13 22:21:24 +03:00
|
|
|
bug.staging.Append(op)
|
2018-07-13 17:13:40 +03:00
|
|
|
}
|
|
|
|
|
2018-07-13 22:21:24 +03:00
|
|
|
// Write the staging area in Git move the operations to the packs
|
|
|
|
func (bug *Bug) Commit(repo repository.Repo) error {
|
|
|
|
if bug.staging.IsEmpty() {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write the Ops as a Git blob containing the serialized array
|
|
|
|
hash, err := bug.staging.Write(repo)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write a Git tree referencing this blob
|
|
|
|
hash, err = repo.StoreTree(map[string]util.Hash{
|
|
|
|
"ops": hash,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write a Git commit referencing the tree, with the previous commit as parent
|
|
|
|
if bug.lastCommit != "" {
|
|
|
|
hash, err = repo.StoreCommitWithParent(hash, bug.lastCommit)
|
|
|
|
} else {
|
|
|
|
hash, err = repo.StoreCommit(hash)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create or update the Git reference for this bug
|
|
|
|
ref := fmt.Sprintf("%s%s", BugsRefPattern, bug.id.String())
|
|
|
|
err = repo.UpdateRef(ref, hash)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
bug.packs = append(bug.packs, bug.staging)
|
|
|
|
bug.staging = OperationPack{}
|
|
|
|
|
|
|
|
return nil
|
2018-07-13 17:13:40 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func (bug *Bug) HumanId() string {
|
2018-07-13 22:21:24 +03:00
|
|
|
return bug.id.String()
|
2018-07-13 17:13:40 +03:00
|
|
|
}
|
2018-07-13 17:48:55 +03:00
|
|
|
|
|
|
|
func (bug *Bug) firstOp() Operation {
|
2018-07-13 22:21:24 +03:00
|
|
|
for _, pack := range bug.packs {
|
2018-07-13 17:48:55 +03:00
|
|
|
for _, op := range pack.Operations {
|
|
|
|
return op
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-13 22:21:24 +03:00
|
|
|
if !bug.staging.IsEmpty() {
|
|
|
|
return bug.staging.Operations[0]
|
2018-07-13 17:48:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|