2018-11-21 20:56:12 +03:00
|
|
|
package identity
|
|
|
|
|
|
|
|
import (
|
2019-01-19 21:23:31 +03:00
|
|
|
"crypto/sha256"
|
2018-11-21 20:56:12 +03:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
|
2019-08-11 15:08:03 +03:00
|
|
|
"github.com/MichaelMure/git-bug/entity"
|
2019-01-19 21:23:31 +03:00
|
|
|
"github.com/MichaelMure/git-bug/repository"
|
2018-11-21 20:56:12 +03:00
|
|
|
"github.com/MichaelMure/git-bug/util/lamport"
|
|
|
|
"github.com/MichaelMure/git-bug/util/text"
|
2019-02-25 01:05:03 +03:00
|
|
|
"github.com/MichaelMure/git-bug/util/timestamp"
|
2018-11-21 20:56:12 +03:00
|
|
|
)
|
|
|
|
|
2019-01-19 21:23:31 +03:00
|
|
|
var _ Interface = &Bare{}
|
2019-08-11 15:08:03 +03:00
|
|
|
var _ entity.Interface = &Bare{}
|
2019-01-19 21:23:31 +03:00
|
|
|
|
2019-01-16 23:23:49 +03:00
|
|
|
// Bare is a very minimal identity, designed to be fully embedded directly along
|
|
|
|
// other data.
|
|
|
|
//
|
|
|
|
// in particular, this identity is designed to be compatible with the handling of
|
|
|
|
// identities in the early version of git-bug.
|
2020-02-15 17:39:49 +03:00
|
|
|
// Deprecated: legacy identity for compat, might make sense to ditch entirely for
|
|
|
|
// simplicity but that would be a breaking change.
|
2018-11-21 20:56:12 +03:00
|
|
|
type Bare struct {
|
2019-08-11 15:08:03 +03:00
|
|
|
id entity.Id
|
2018-11-21 20:56:12 +03:00
|
|
|
name string
|
|
|
|
email string
|
2020-02-25 23:35:57 +03:00
|
|
|
login string
|
2018-11-21 20:56:12 +03:00
|
|
|
avatarUrl string
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewBare(name string, email string) *Bare {
|
2019-08-11 15:08:03 +03:00
|
|
|
return &Bare{id: entity.UnsetId, name: name, email: email}
|
2018-11-21 20:56:12 +03:00
|
|
|
}
|
|
|
|
|
2020-02-25 23:35:57 +03:00
|
|
|
func NewBareFull(name string, email string, login string, avatarUrl string) *Bare {
|
|
|
|
return &Bare{id: entity.UnsetId, name: name, email: email, login: login, avatarUrl: avatarUrl}
|
2019-08-11 15:08:03 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func deriveId(data []byte) entity.Id {
|
|
|
|
sum := sha256.Sum256(data)
|
|
|
|
return entity.Id(fmt.Sprintf("%x", sum))
|
2018-11-21 20:56:12 +03:00
|
|
|
}
|
|
|
|
|
2019-01-20 17:41:27 +03:00
|
|
|
type bareIdentityJSON struct {
|
2018-11-21 20:56:12 +03:00
|
|
|
Name string `json:"name,omitempty"`
|
|
|
|
Email string `json:"email,omitempty"`
|
2020-02-25 23:35:57 +03:00
|
|
|
Login string `json:"login,omitempty"`
|
2018-11-21 20:56:12 +03:00
|
|
|
AvatarUrl string `json:"avatar_url,omitempty"`
|
|
|
|
}
|
|
|
|
|
2019-01-19 21:23:31 +03:00
|
|
|
func (i *Bare) MarshalJSON() ([]byte, error) {
|
2019-08-12 17:12:14 +03:00
|
|
|
return json.Marshal(bareIdentityJSON{
|
2018-11-21 20:56:12 +03:00
|
|
|
Name: i.name,
|
|
|
|
Email: i.email,
|
2020-02-25 23:35:57 +03:00
|
|
|
Login: i.login,
|
2018-11-21 20:56:12 +03:00
|
|
|
AvatarUrl: i.avatarUrl,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-01-19 21:23:31 +03:00
|
|
|
func (i *Bare) UnmarshalJSON(data []byte) error {
|
2019-08-11 15:08:03 +03:00
|
|
|
// Compute the Id when loading the op from disk.
|
|
|
|
i.id = deriveId(data)
|
|
|
|
|
2019-01-20 17:41:27 +03:00
|
|
|
aux := bareIdentityJSON{}
|
2018-11-21 20:56:12 +03:00
|
|
|
|
|
|
|
if err := json.Unmarshal(data, &aux); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
i.name = aux.Name
|
|
|
|
i.email = aux.Email
|
2020-02-25 23:35:57 +03:00
|
|
|
i.login = aux.Login
|
2018-11-21 20:56:12 +03:00
|
|
|
i.avatarUrl = aux.AvatarUrl
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-02-19 02:19:27 +03:00
|
|
|
// Id return the Identity identifier
|
2019-08-11 15:08:03 +03:00
|
|
|
func (i *Bare) Id() entity.Id {
|
|
|
|
// We don't have a proper Id at hand, so let's hash all the data to get one.
|
2019-01-19 21:23:31 +03:00
|
|
|
|
2019-08-11 15:08:03 +03:00
|
|
|
if i.id == "" {
|
|
|
|
// something went really wrong
|
|
|
|
panic("identity's id not set")
|
2019-01-19 21:23:31 +03:00
|
|
|
}
|
2019-08-11 15:08:03 +03:00
|
|
|
if i.id == entity.UnsetId {
|
|
|
|
// This means we are trying to get the identity identifier *before* it has been stored
|
|
|
|
// As the Id is computed based on the actual bytes written on the disk, we are going to predict
|
|
|
|
// those and then get the Id. This is safe as it will be the exact same code writing on disk later.
|
2019-01-19 21:23:31 +03:00
|
|
|
|
2019-08-11 15:08:03 +03:00
|
|
|
data, err := json.Marshal(i)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2019-01-19 21:23:31 +03:00
|
|
|
|
2019-08-11 15:08:03 +03:00
|
|
|
i.id = deriveId(data)
|
|
|
|
}
|
2019-01-19 21:23:31 +03:00
|
|
|
return i.id
|
|
|
|
}
|
|
|
|
|
2019-02-19 02:19:27 +03:00
|
|
|
// Name return the last version of the name
|
2019-01-19 21:23:31 +03:00
|
|
|
func (i *Bare) Name() string {
|
2018-11-21 20:56:12 +03:00
|
|
|
return i.name
|
|
|
|
}
|
|
|
|
|
2019-02-19 02:19:27 +03:00
|
|
|
// Email return the last version of the email
|
2019-01-19 21:23:31 +03:00
|
|
|
func (i *Bare) Email() string {
|
2018-11-21 20:56:12 +03:00
|
|
|
return i.email
|
|
|
|
}
|
|
|
|
|
2020-02-25 23:35:57 +03:00
|
|
|
// Login return the last version of the login
|
|
|
|
func (i *Bare) Login() string {
|
|
|
|
return i.login
|
|
|
|
}
|
|
|
|
|
2019-02-19 02:19:27 +03:00
|
|
|
// AvatarUrl return the last version of the Avatar URL
|
2019-01-19 21:23:31 +03:00
|
|
|
func (i *Bare) AvatarUrl() string {
|
2018-11-21 20:56:12 +03:00
|
|
|
return i.avatarUrl
|
|
|
|
}
|
|
|
|
|
2019-01-16 23:23:49 +03:00
|
|
|
// Keys return the last version of the valid keys
|
2020-01-24 02:30:13 +03:00
|
|
|
func (i *Bare) Keys() []*Key {
|
|
|
|
return nil
|
2018-11-21 20:56:12 +03:00
|
|
|
}
|
|
|
|
|
2019-01-16 23:23:49 +03:00
|
|
|
// ValidKeysAtTime return the set of keys valid at a given lamport time
|
2020-01-24 02:30:13 +03:00
|
|
|
func (i *Bare) ValidKeysAtTime(_ lamport.Time) []*Key {
|
|
|
|
return nil
|
2018-11-21 20:56:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// DisplayName return a non-empty string to display, representing the
|
|
|
|
// identity, based on the non-empty values.
|
2019-01-19 21:23:31 +03:00
|
|
|
func (i *Bare) DisplayName() string {
|
2020-02-25 23:35:57 +03:00
|
|
|
switch {
|
|
|
|
case i.name == "" && i.login != "":
|
|
|
|
return i.login
|
|
|
|
case i.name != "" && i.login == "":
|
|
|
|
return i.name
|
|
|
|
case i.name != "" && i.login != "":
|
|
|
|
return fmt.Sprintf("%s (%s)", i.name, i.login)
|
|
|
|
}
|
|
|
|
|
|
|
|
panic("invalid person data")
|
2018-11-21 20:56:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Validate check if the Identity data is valid
|
2019-01-19 21:23:31 +03:00
|
|
|
func (i *Bare) Validate() error {
|
2020-02-25 23:35:57 +03:00
|
|
|
if text.Empty(i.name) && text.Empty(i.login) {
|
|
|
|
return fmt.Errorf("either name or login should be set")
|
2018-11-21 20:56:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if strings.Contains(i.name, "\n") {
|
|
|
|
return fmt.Errorf("name should be a single line")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !text.Safe(i.name) {
|
|
|
|
return fmt.Errorf("name is not fully printable")
|
|
|
|
}
|
|
|
|
|
2020-02-25 23:35:57 +03:00
|
|
|
if strings.Contains(i.login, "\n") {
|
|
|
|
return fmt.Errorf("login should be a single line")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !text.Safe(i.login) {
|
|
|
|
return fmt.Errorf("login is not fully printable")
|
|
|
|
}
|
|
|
|
|
2018-11-21 20:56:12 +03:00
|
|
|
if strings.Contains(i.email, "\n") {
|
|
|
|
return fmt.Errorf("email should be a single line")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !text.Safe(i.email) {
|
|
|
|
return fmt.Errorf("email is not fully printable")
|
|
|
|
}
|
|
|
|
|
|
|
|
if i.avatarUrl != "" && !text.ValidUrl(i.avatarUrl) {
|
|
|
|
return fmt.Errorf("avatarUrl is not a valid URL")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-01-19 21:23:31 +03:00
|
|
|
// Write the identity into the Repository. In particular, this ensure that
|
|
|
|
// the Id is properly set.
|
2019-02-19 02:19:27 +03:00
|
|
|
func (i *Bare) Commit(repo repository.ClockedRepo) error {
|
2019-01-19 21:23:31 +03:00
|
|
|
// Nothing to do, everything is directly embedded
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-02-19 02:19:27 +03:00
|
|
|
// If needed, write the identity into the Repository. In particular, this
|
|
|
|
// ensure that the Id is properly set.
|
|
|
|
func (i *Bare) CommitAsNeeded(repo repository.ClockedRepo) error {
|
2019-02-16 19:32:30 +03:00
|
|
|
// Nothing to do, everything is directly embedded
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-01-16 23:23:49 +03:00
|
|
|
// IsProtected return true if the chain of git commits started to be signed.
|
|
|
|
// If that's the case, only signed commit with a valid key for this identity can be added.
|
2019-01-19 21:23:31 +03:00
|
|
|
func (i *Bare) IsProtected() bool {
|
2018-11-21 20:56:12 +03:00
|
|
|
return false
|
|
|
|
}
|
2019-02-25 01:05:03 +03:00
|
|
|
|
|
|
|
// LastModificationLamportTime return the Lamport time at which the last version of the identity became valid.
|
|
|
|
func (i *Bare) LastModificationLamport() lamport.Time {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
// LastModification return the timestamp at which the last version of the identity became valid.
|
|
|
|
func (i *Bare) LastModification() timestamp.Timestamp {
|
|
|
|
return 0
|
|
|
|
}
|