| // Copyright 2019 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package sumdb |
| |
| import ( |
| "context" |
| "fmt" |
| "sync" |
| |
| "golang.org/x/mod/module" |
| "golang.org/x/mod/sumdb/note" |
| "golang.org/x/mod/sumdb/tlog" |
| ) |
| |
| // NewTestServer constructs a new [TestServer] |
| // that will sign its tree with the given signer key |
| // (see [golang.org/x/mod/sumdb/note]) |
| // and fetch new records as needed by calling gosum. |
| func NewTestServer(signer string, gosum func(path, vers string) ([]byte, error)) *TestServer { |
| return &TestServer{signer: signer, gosum: gosum} |
| } |
| |
| // A TestServer is an in-memory implementation of [ServerOps] for testing. |
| type TestServer struct { |
| signer string |
| gosum func(path, vers string) ([]byte, error) |
| |
| mu sync.Mutex |
| hashes testHashes |
| records [][]byte |
| lookup map[string]int64 |
| } |
| |
| // testHashes implements tlog.HashReader, reading from a slice. |
| type testHashes []tlog.Hash |
| |
| func (h testHashes) ReadHashes(indexes []int64) ([]tlog.Hash, error) { |
| var list []tlog.Hash |
| for _, id := range indexes { |
| list = append(list, h[id]) |
| } |
| return list, nil |
| } |
| |
| func (s *TestServer) Signed(ctx context.Context) ([]byte, error) { |
| s.mu.Lock() |
| defer s.mu.Unlock() |
| |
| size := int64(len(s.records)) |
| h, err := tlog.TreeHash(size, s.hashes) |
| if err != nil { |
| return nil, err |
| } |
| text := tlog.FormatTree(tlog.Tree{N: size, Hash: h}) |
| signer, err := note.NewSigner(s.signer) |
| if err != nil { |
| return nil, err |
| } |
| return note.Sign(¬e.Note{Text: string(text)}, signer) |
| } |
| |
| func (s *TestServer) ReadRecords(ctx context.Context, id, n int64) ([][]byte, error) { |
| s.mu.Lock() |
| defer s.mu.Unlock() |
| |
| var list [][]byte |
| for i := int64(0); i < n; i++ { |
| if id+i >= int64(len(s.records)) { |
| return nil, fmt.Errorf("missing records") |
| } |
| list = append(list, s.records[id+i]) |
| } |
| return list, nil |
| } |
| |
| func (s *TestServer) Lookup(ctx context.Context, m module.Version) (int64, error) { |
| key := m.String() |
| s.mu.Lock() |
| id, ok := s.lookup[key] |
| s.mu.Unlock() |
| if ok { |
| return id, nil |
| } |
| |
| // Look up module and compute go.sum lines. |
| data, err := s.gosum(m.Path, m.Version) |
| if err != nil { |
| return 0, err |
| } |
| |
| s.mu.Lock() |
| defer s.mu.Unlock() |
| |
| // We ran the fetch without the lock. |
| // If another fetch happened and committed, use it instead. |
| id, ok = s.lookup[key] |
| if ok { |
| return id, nil |
| } |
| |
| // Add record. |
| id = int64(len(s.records)) |
| s.records = append(s.records, data) |
| if s.lookup == nil { |
| s.lookup = make(map[string]int64) |
| } |
| s.lookup[key] = id |
| hashes, err := tlog.StoredHashesForRecordHash(id, tlog.RecordHash(data), s.hashes) |
| if err != nil { |
| panic(err) |
| } |
| s.hashes = append(s.hashes, hashes...) |
| |
| return id, nil |
| } |
| |
| func (s *TestServer) ReadTileData(ctx context.Context, t tlog.Tile) ([]byte, error) { |
| s.mu.Lock() |
| defer s.mu.Unlock() |
| |
| return tlog.ReadTileData(t, s.hashes) |
| } |