cmd/hakurei/parse: use new store interface
	
		
			
	
		
	
	
		
	
		
			All checks were successful
		
		
	
	
		
			
				
	
				Test / Create distribution (push) Successful in 35s
				
			
		
			
				
	
				Test / Sandbox (push) Successful in 2m21s
				
			
		
			
				
	
				Test / Sandbox (race detector) (push) Successful in 4m16s
				
			
		
			
				
	
				Test / Hpkg (push) Successful in 4m15s
				
			
		
			
				
	
				Test / Hakurei (race detector) (push) Successful in 4m58s
				
			
		
			
				
	
				Test / Hakurei (push) Successful in 2m16s
				
			
		
			
				
	
				Test / Flake checks (push) Successful in 1m28s
				
			
		
		
	
	
				
					
				
			
		
			All checks were successful
		
		
	
	Test / Create distribution (push) Successful in 35s
				
			Test / Sandbox (push) Successful in 2m21s
				
			Test / Sandbox (race detector) (push) Successful in 4m16s
				
			Test / Hpkg (push) Successful in 4m15s
				
			Test / Hakurei (race detector) (push) Successful in 4m58s
				
			Test / Hakurei (push) Successful in 2m16s
				
			Test / Flake checks (push) Successful in 1m28s
				
			This greatly reduces overhead. The iterator also significantly cleans up the usage code. Signed-off-by: Ophestra <cat@gensokyo.uk>
This commit is contained in:
		
							parent
							
								
									898b5aed3d
								
							
						
					
					
						commit
						23ae7822bf
					
				@ -294,7 +294,10 @@ func buildCommand(ctx context.Context, msg message.Msg, early *earlyHardeningErr
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	{
 | 
						{
 | 
				
			||||||
		var flagShort bool
 | 
							var (
 | 
				
			||||||
 | 
								flagShort   bool
 | 
				
			||||||
 | 
								flagNoStore bool
 | 
				
			||||||
 | 
							)
 | 
				
			||||||
		c.NewCommand("show", "Show live or local app configuration", func(args []string) error {
 | 
							c.NewCommand("show", "Show live or local app configuration", func(args []string) error {
 | 
				
			||||||
			switch len(args) {
 | 
								switch len(args) {
 | 
				
			||||||
			case 0: // system
 | 
								case 0: // system
 | 
				
			||||||
@ -302,10 +305,23 @@ func buildCommand(ctx context.Context, msg message.Msg, early *earlyHardeningErr
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
			case 1: // instance
 | 
								case 1: // instance
 | 
				
			||||||
				name := args[0]
 | 
									name := args[0]
 | 
				
			||||||
				config, entry := tryIdentifier(msg, name)
 | 
					
 | 
				
			||||||
				if config == nil {
 | 
									var (
 | 
				
			||||||
					config = tryPath(msg, name)
 | 
										config *hst.Config
 | 
				
			||||||
 | 
										entry  *hst.State
 | 
				
			||||||
 | 
									)
 | 
				
			||||||
 | 
									if !flagNoStore {
 | 
				
			||||||
 | 
										var sc hst.Paths
 | 
				
			||||||
 | 
										env.CopyPaths().Copy(&sc, new(outcome.Hsu).MustID(nil))
 | 
				
			||||||
 | 
										entry = tryIdentifier(msg, name, outcome.NewStore(&sc))
 | 
				
			||||||
				}
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									if entry == nil {
 | 
				
			||||||
 | 
										config = tryPath(msg, name)
 | 
				
			||||||
 | 
									} else {
 | 
				
			||||||
 | 
										config = entry.Config
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
				if !printShowInstance(os.Stdout, time.Now().UTC(), entry, config, flagShort, flagJSON) {
 | 
									if !printShowInstance(os.Stdout, time.Now().UTC(), entry, config, flagShort, flagJSON) {
 | 
				
			||||||
					os.Exit(1)
 | 
										os.Exit(1)
 | 
				
			||||||
				}
 | 
									}
 | 
				
			||||||
@ -314,7 +330,9 @@ func buildCommand(ctx context.Context, msg message.Msg, early *earlyHardeningErr
 | 
				
			|||||||
				log.Fatal("show requires 1 argument")
 | 
									log.Fatal("show requires 1 argument")
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
			return errSuccess
 | 
								return errSuccess
 | 
				
			||||||
		}).Flag(&flagShort, "short", command.BoolFlag(false), "Omit filesystem information")
 | 
							}).
 | 
				
			||||||
 | 
								Flag(&flagShort, "short", command.BoolFlag(false), "Omit filesystem information").
 | 
				
			||||||
 | 
								Flag(&flagNoStore, "no-store", command.BoolFlag(false), "Do not attempt to match from active instances")
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	{
 | 
						{
 | 
				
			||||||
 | 
				
			|||||||
@ -11,8 +11,6 @@ import (
 | 
				
			|||||||
	"syscall"
 | 
						"syscall"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	"hakurei.app/hst"
 | 
						"hakurei.app/hst"
 | 
				
			||||||
	"hakurei.app/internal/env"
 | 
					 | 
				
			||||||
	"hakurei.app/internal/outcome"
 | 
					 | 
				
			||||||
	"hakurei.app/internal/store"
 | 
						"hakurei.app/internal/store"
 | 
				
			||||||
	"hakurei.app/message"
 | 
						"hakurei.app/message"
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
@ -81,26 +79,7 @@ func shortIdentifierString(s string) string {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// tryIdentifier attempts to match [hst.State] from a [hex] representation of [hst.ID] or a prefix of its lower half.
 | 
					// tryIdentifier attempts to match [hst.State] from a [hex] representation of [hst.ID] or a prefix of its lower half.
 | 
				
			||||||
func tryIdentifier(msg message.Msg, name string) (config *hst.Config, entry *hst.State) {
 | 
					func tryIdentifier(msg message.Msg, name string, s *store.Store) *hst.State {
 | 
				
			||||||
	return tryIdentifierEntries(msg, name, func() map[hst.ID]*hst.State {
 | 
					 | 
				
			||||||
		var sc hst.Paths
 | 
					 | 
				
			||||||
		env.CopyPaths().Copy(&sc, new(outcome.Hsu).MustID(nil))
 | 
					 | 
				
			||||||
		s := store.NewMulti(msg, sc.SharePath)
 | 
					 | 
				
			||||||
		if entries, err := store.Join(s); err != nil {
 | 
					 | 
				
			||||||
			msg.GetLogger().Printf("cannot join store: %v", err) // not fatal
 | 
					 | 
				
			||||||
			return nil
 | 
					 | 
				
			||||||
		} else {
 | 
					 | 
				
			||||||
			return entries
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
	})
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// tryIdentifierEntries implements tryIdentifier with a custom entries pair getter.
 | 
					 | 
				
			||||||
func tryIdentifierEntries(
 | 
					 | 
				
			||||||
	msg message.Msg,
 | 
					 | 
				
			||||||
	name string,
 | 
					 | 
				
			||||||
	getEntries func() map[hst.ID]*hst.State,
 | 
					 | 
				
			||||||
) (config *hst.Config, entry *hst.State) {
 | 
					 | 
				
			||||||
	const (
 | 
						const (
 | 
				
			||||||
		likeShort = 1 << iota
 | 
							likeShort = 1 << iota
 | 
				
			||||||
		likeFull
 | 
							likeFull
 | 
				
			||||||
@ -116,7 +95,7 @@ func tryIdentifierEntries(
 | 
				
			|||||||
			if c >= 'a' && c <= 'f' {
 | 
								if c >= 'a' && c <= 'f' {
 | 
				
			||||||
				continue
 | 
									continue
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
			return
 | 
								return nil
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		likely |= likeShort
 | 
							likely |= likeShort
 | 
				
			||||||
	} else if len(name) == hex.EncodedLen(len(hst.ID{})) {
 | 
						} else if len(name) == hex.EncodedLen(len(hst.ID{})) {
 | 
				
			||||||
@ -124,40 +103,58 @@ func tryIdentifierEntries(
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if likely == 0 {
 | 
						if likely == 0 {
 | 
				
			||||||
		return
 | 
							return nil
 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	entries := getEntries()
 | 
					 | 
				
			||||||
	if entries == nil {
 | 
					 | 
				
			||||||
		return
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						entries, copyError := s.All()
 | 
				
			||||||
 | 
						defer func() {
 | 
				
			||||||
 | 
							if err := copyError(); err != nil {
 | 
				
			||||||
 | 
								msg.GetLogger().Println(getMessage("cannot iterate over store:", err))
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	switch {
 | 
						switch {
 | 
				
			||||||
	case likely&likeShort != 0:
 | 
						case likely&likeShort != 0:
 | 
				
			||||||
		msg.Verbose("argument looks like short identifier")
 | 
							msg.Verbose("argument looks like short identifier")
 | 
				
			||||||
		for id := range entries {
 | 
							for eh := range entries {
 | 
				
			||||||
			v := id.String()
 | 
								if eh.DecodeErr != nil {
 | 
				
			||||||
			if strings.HasPrefix(v[len(hst.ID{}):], name) {
 | 
									msg.Verbose(getMessage("skipping instance:", eh.DecodeErr))
 | 
				
			||||||
				// match, use config from this state entry
 | 
									continue
 | 
				
			||||||
				entry = entries[id]
 | 
					 | 
				
			||||||
				config = entry.Config
 | 
					 | 
				
			||||||
				break
 | 
					 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			msg.Verbosef("instance %s skipped", v)
 | 
								if strings.HasPrefix(eh.ID.String()[len(hst.ID{}):], name) {
 | 
				
			||||||
 | 
									var entry hst.State
 | 
				
			||||||
 | 
									if _, err := eh.Load(&entry); err != nil {
 | 
				
			||||||
 | 
										msg.GetLogger().Println(getMessage("cannot load state entry:", err))
 | 
				
			||||||
 | 
										continue
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
									return &entry
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		return
 | 
							return nil
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	case likely&likeFull != 0:
 | 
						case likely&likeFull != 0:
 | 
				
			||||||
		var likelyID hst.ID
 | 
							var likelyID hst.ID
 | 
				
			||||||
		if likelyID.UnmarshalText([]byte(name)) != nil {
 | 
							if likelyID.UnmarshalText([]byte(name)) != nil {
 | 
				
			||||||
			return
 | 
								return nil
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		msg.Verbose("argument looks like identifier")
 | 
							msg.Verbose("argument looks like identifier")
 | 
				
			||||||
		if ent, ok := entries[likelyID]; ok {
 | 
							for eh := range entries {
 | 
				
			||||||
			entry = ent
 | 
								if eh.DecodeErr != nil {
 | 
				
			||||||
			config = ent.Config
 | 
									msg.Verbose(getMessage("skipping instance:", eh.DecodeErr))
 | 
				
			||||||
 | 
									continue
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								if eh.ID == likelyID {
 | 
				
			||||||
 | 
									var entry hst.State
 | 
				
			||||||
 | 
									if _, err := eh.Load(&entry); err != nil {
 | 
				
			||||||
 | 
										msg.GetLogger().Println(getMessage("cannot load state entry:", err))
 | 
				
			||||||
 | 
										continue
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
									return &entry
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		return
 | 
							return nil
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	default:
 | 
						default:
 | 
				
			||||||
		panic("unreachable")
 | 
							panic("unreachable")
 | 
				
			||||||
 | 
				
			|||||||
@ -1,10 +1,14 @@
 | 
				
			|||||||
package main
 | 
					package main
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import (
 | 
					import (
 | 
				
			||||||
 | 
						"bytes"
 | 
				
			||||||
	"reflect"
 | 
						"reflect"
 | 
				
			||||||
	"testing"
 | 
						"testing"
 | 
				
			||||||
 | 
						"time"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"hakurei.app/container/check"
 | 
				
			||||||
	"hakurei.app/hst"
 | 
						"hakurei.app/hst"
 | 
				
			||||||
 | 
						"hakurei.app/internal/store"
 | 
				
			||||||
	"hakurei.app/message"
 | 
						"hakurei.app/message"
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -23,17 +27,47 @@ func TestShortIdentifier(t *testing.T) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
func TestTryIdentifier(t *testing.T) {
 | 
					func TestTryIdentifier(t *testing.T) {
 | 
				
			||||||
	t.Parallel()
 | 
						t.Parallel()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	msg := message.NewMsg(nil)
 | 
						msg := message.NewMsg(nil)
 | 
				
			||||||
	id := hst.ID{
 | 
						id := hst.ID{
 | 
				
			||||||
		0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef,
 | 
							0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef,
 | 
				
			||||||
		0xfe, 0xdc, 0xba, 0x98, 0x76, 0x54, 0x32, 0x10,
 | 
							0xfe, 0xdc, 0xba, 0x98, 0x76, 0x54, 0x32, 0x10,
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
						withBase := func(extra ...hst.State) []hst.State {
 | 
				
			||||||
 | 
							return append([]hst.State{
 | 
				
			||||||
 | 
								{ID: (hst.ID)(bytes.Repeat([]byte{0xaa}, len(hst.ID{}))), PID: 0xbeef, ShimPID: 0xcafe, Config: hst.Template(), Time: time.Unix(0, 0xdeadbeef0)},
 | 
				
			||||||
 | 
								{ID: (hst.ID)(bytes.Repeat([]byte{0xab}, len(hst.ID{}))), PID: 0x1beef, ShimPID: 0x1cafe, Config: hst.Template(), Time: time.Unix(0, 0xdeadbeef1)},
 | 
				
			||||||
 | 
								{ID: (hst.ID)(bytes.Repeat([]byte{0xf0}, len(hst.ID{}))), PID: 0x2beef, ShimPID: 0x2cafe, Config: hst.Template(), Time: time.Unix(0, 0xdeadbeef2)},
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								{ID: (hst.ID)(bytes.Repeat([]byte{0xfe}, len(hst.ID{}))), PID: 0xbed, ShimPID: 0xfff, Config: func() *hst.Config {
 | 
				
			||||||
 | 
									template := hst.Template()
 | 
				
			||||||
 | 
									template.Identity = hst.IdentityMax
 | 
				
			||||||
 | 
									return template
 | 
				
			||||||
 | 
								}(), Time: time.Unix(0, 0xcafebabe0)},
 | 
				
			||||||
 | 
								{ID: (hst.ID)(bytes.Repeat([]byte{0xfc}, len(hst.ID{}))), PID: 0x1bed, ShimPID: 0x1fff, Config: func() *hst.Config {
 | 
				
			||||||
 | 
									template := hst.Template()
 | 
				
			||||||
 | 
									template.Identity = 0xfc
 | 
				
			||||||
 | 
									return template
 | 
				
			||||||
 | 
								}(), Time: time.Unix(0, 0xcafebabe1)},
 | 
				
			||||||
 | 
								{ID: (hst.ID)(bytes.Repeat([]byte{0xce}, len(hst.ID{}))), PID: 0x2bed, ShimPID: 0x2fff, Config: func() *hst.Config {
 | 
				
			||||||
 | 
									template := hst.Template()
 | 
				
			||||||
 | 
									template.Identity = 0xce
 | 
				
			||||||
 | 
									return template
 | 
				
			||||||
 | 
								}(), Time: time.Unix(0, 0xcafebabe2)},
 | 
				
			||||||
 | 
							}, extra...)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						sampleEntry := hst.State{
 | 
				
			||||||
 | 
							ID:      id,
 | 
				
			||||||
 | 
							PID:     0xcafebabe,
 | 
				
			||||||
 | 
							ShimPID: 0xdeadbeef,
 | 
				
			||||||
 | 
							Config:  hst.Template(),
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	testCases := []struct {
 | 
						testCases := []struct {
 | 
				
			||||||
		name    string
 | 
							name string
 | 
				
			||||||
		s       string
 | 
							s    string
 | 
				
			||||||
		entries map[hst.ID]*hst.State
 | 
							data []hst.State
 | 
				
			||||||
		want    *hst.State
 | 
							want *hst.State
 | 
				
			||||||
	}{
 | 
						}{
 | 
				
			||||||
		{"likely entries fault", "ffffffff", nil, nil},
 | 
							{"likely entries fault", "ffffffff", nil, nil},
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -41,58 +75,40 @@ func TestTryIdentifier(t *testing.T) {
 | 
				
			|||||||
		{"likely short too long", "fffffffffffffffff", nil, nil},
 | 
							{"likely short too long", "fffffffffffffffff", nil, nil},
 | 
				
			||||||
		{"likely short invalid lower", "fffffff\x00", nil, nil},
 | 
							{"likely short invalid lower", "fffffff\x00", nil, nil},
 | 
				
			||||||
		{"likely short invalid higher", "0000000\xff", nil, nil},
 | 
							{"likely short invalid higher", "0000000\xff", nil, nil},
 | 
				
			||||||
		{"short no match", "fedcba98", map[hst.ID]*hst.State{hst.ID{}: nil}, nil},
 | 
							{"short no match", "fedcba98", withBase(), nil},
 | 
				
			||||||
		{"short match", "fedcba98", map[hst.ID]*hst.State{
 | 
							{"short match", "fedcba98", withBase(sampleEntry), &sampleEntry},
 | 
				
			||||||
			hst.ID{}: nil,
 | 
							{"short match single", "fedcba98", []hst.State{sampleEntry}, &sampleEntry},
 | 
				
			||||||
			id: {
 | 
							{"short match longer", "fedcba98765", withBase(sampleEntry), &sampleEntry},
 | 
				
			||||||
				ID:      id,
 | 
					 | 
				
			||||||
				PID:     0xcafebabe,
 | 
					 | 
				
			||||||
				ShimPID: 0xdeadbeef,
 | 
					 | 
				
			||||||
				Config:  hst.Template(),
 | 
					 | 
				
			||||||
			},
 | 
					 | 
				
			||||||
		}, &hst.State{
 | 
					 | 
				
			||||||
			ID:      id,
 | 
					 | 
				
			||||||
			PID:     0xcafebabe,
 | 
					 | 
				
			||||||
			ShimPID: 0xdeadbeef,
 | 
					 | 
				
			||||||
			Config:  hst.Template(),
 | 
					 | 
				
			||||||
		}},
 | 
					 | 
				
			||||||
		{"short match longer", "fedcba98765", map[hst.ID]*hst.State{
 | 
					 | 
				
			||||||
			hst.ID{}: nil,
 | 
					 | 
				
			||||||
			id: {
 | 
					 | 
				
			||||||
				ID:      id,
 | 
					 | 
				
			||||||
				PID:     0xcafebabe,
 | 
					 | 
				
			||||||
				ShimPID: 0xdeadbeef,
 | 
					 | 
				
			||||||
				Config:  hst.Template(),
 | 
					 | 
				
			||||||
			},
 | 
					 | 
				
			||||||
		}, &hst.State{
 | 
					 | 
				
			||||||
			ID:      id,
 | 
					 | 
				
			||||||
			PID:     0xcafebabe,
 | 
					 | 
				
			||||||
			ShimPID: 0xdeadbeef,
 | 
					 | 
				
			||||||
			Config:  hst.Template(),
 | 
					 | 
				
			||||||
		}},
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
		{"likely long invalid", "0123456789abcdeffedcba987654321\x00", map[hst.ID]*hst.State{}, nil},
 | 
							{"likely long invalid", "0123456789abcdeffedcba987654321\x00", nil, nil},
 | 
				
			||||||
		{"long no match", "0123456789abcdeffedcba9876543210", map[hst.ID]*hst.State{hst.ID{}: nil}, nil},
 | 
							{"long no match", "0123456789abcdeffedcba9876543210", withBase(), nil},
 | 
				
			||||||
		{"long match", "0123456789abcdeffedcba9876543210", map[hst.ID]*hst.State{
 | 
							{"long match", "0123456789abcdeffedcba9876543210", withBase(sampleEntry), &sampleEntry},
 | 
				
			||||||
			hst.ID{}: nil,
 | 
							{"long match single", "0123456789abcdeffedcba9876543210", []hst.State{sampleEntry}, &sampleEntry},
 | 
				
			||||||
			id: {
 | 
					 | 
				
			||||||
				ID:      id,
 | 
					 | 
				
			||||||
				PID:     0xcafebabe,
 | 
					 | 
				
			||||||
				ShimPID: 0xdeadbeef,
 | 
					 | 
				
			||||||
				Config:  hst.Template(),
 | 
					 | 
				
			||||||
			},
 | 
					 | 
				
			||||||
		}, &hst.State{
 | 
					 | 
				
			||||||
			ID:      id,
 | 
					 | 
				
			||||||
			PID:     0xcafebabe,
 | 
					 | 
				
			||||||
			ShimPID: 0xdeadbeef,
 | 
					 | 
				
			||||||
			Config:  hst.Template(),
 | 
					 | 
				
			||||||
		}},
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	for _, tc := range testCases {
 | 
						for _, tc := range testCases {
 | 
				
			||||||
 | 
							base := check.MustAbs(t.TempDir()).Append("store")
 | 
				
			||||||
 | 
							s := store.New(base)
 | 
				
			||||||
 | 
							for i := range tc.data {
 | 
				
			||||||
 | 
								if h, err := s.Handle(tc.data[i].Identity); err != nil {
 | 
				
			||||||
 | 
									t.Fatalf("Handle: error = %v", err)
 | 
				
			||||||
 | 
								} else {
 | 
				
			||||||
 | 
									var unlock func()
 | 
				
			||||||
 | 
									if unlock, err = h.Lock(); err != nil {
 | 
				
			||||||
 | 
										t.Fatalf("Lock: error = %v", err)
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
									_, err = h.Save(&tc.data[i])
 | 
				
			||||||
 | 
									unlock()
 | 
				
			||||||
 | 
									if err != nil {
 | 
				
			||||||
 | 
										t.Fatalf("Save: error = %v", err)
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// store must not be written to beyond this point
 | 
				
			||||||
		t.Run(tc.name, func(t *testing.T) {
 | 
							t.Run(tc.name, func(t *testing.T) {
 | 
				
			||||||
			t.Parallel()
 | 
								t.Parallel()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			_, got := tryIdentifierEntries(msg, tc.s, func() map[hst.ID]*hst.State { return tc.entries })
 | 
								got := tryIdentifier(msg, tc.s, store.New(base))
 | 
				
			||||||
			if !reflect.DeepEqual(got, tc.want) {
 | 
								if !reflect.DeepEqual(got, tc.want) {
 | 
				
			||||||
				t.Errorf("tryIdentifier: %#v, want %#v", got, tc.want)
 | 
									t.Errorf("tryIdentifier: %#v, want %#v", got, tc.want)
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
 | 
				
			|||||||
@ -287,3 +287,11 @@ func mustPrintln(output io.Writer, a ...any) {
 | 
				
			|||||||
		log.Fatalf("cannot print: %v", err)
 | 
							log.Fatalf("cannot print: %v", err)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// getMessage returns a [message.Error] message if available, or err prefixed with fallback otherwise.
 | 
				
			||||||
 | 
					func getMessage(fallback string, err error) string {
 | 
				
			||||||
 | 
						if m, ok := message.GetMessage(err); ok {
 | 
				
			||||||
 | 
							return m
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return fmt.Sprintln(fallback, err)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -227,7 +227,7 @@ in
 | 
				
			|||||||
                    in
 | 
					                    in
 | 
				
			||||||
                    pkgs.runCommand "checked-${name}" { nativeBuildInputs = [ cfg.package ]; } ''
 | 
					                    pkgs.runCommand "checked-${name}" { nativeBuildInputs = [ cfg.package ]; } ''
 | 
				
			||||||
                      ln -vs ${file} "$out"
 | 
					                      ln -vs ${file} "$out"
 | 
				
			||||||
                      hakurei show ${file}
 | 
					                      hakurei show --no-store ${file}
 | 
				
			||||||
                    '';
 | 
					                    '';
 | 
				
			||||||
                in
 | 
					                in
 | 
				
			||||||
                pkgs.writeShellScriptBin app.name ''
 | 
					                pkgs.writeShellScriptBin app.name ''
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user