1
2
3
4
5 package modfetch
6
7 import (
8 "bytes"
9 "context"
10 "encoding/json"
11 "errors"
12 "fmt"
13 "io"
14 "io/fs"
15 "math/rand"
16 "os"
17 "path/filepath"
18 "strconv"
19 "strings"
20 "sync"
21
22 "cmd/go/internal/base"
23 "cmd/go/internal/cfg"
24 "cmd/go/internal/gover"
25 "cmd/go/internal/lockedfile"
26 "cmd/go/internal/modfetch/codehost"
27 "cmd/internal/par"
28 "cmd/internal/robustio"
29 "cmd/internal/telemetry/counter"
30
31 "golang.org/x/mod/module"
32 "golang.org/x/mod/semver"
33 )
34
35 func cacheDir(ctx context.Context, path string) (string, error) {
36 if err := checkCacheDir(ctx); err != nil {
37 return "", err
38 }
39 enc, err := module.EscapePath(path)
40 if err != nil {
41 return "", err
42 }
43 return filepath.Join(cfg.GOMODCACHE, "cache/download", enc, "/@v"), nil
44 }
45
46 func CachePath(ctx context.Context, m module.Version, suffix string) (string, error) {
47 if gover.IsToolchain(m.Path) {
48 return "", ErrToolchain
49 }
50 dir, err := cacheDir(ctx, m.Path)
51 if err != nil {
52 return "", err
53 }
54 if !gover.ModIsValid(m.Path, m.Version) {
55 return "", fmt.Errorf("non-semver module version %q", m.Version)
56 }
57 if module.CanonicalVersion(m.Version) != m.Version {
58 return "", fmt.Errorf("non-canonical module version %q", m.Version)
59 }
60 encVer, err := module.EscapeVersion(m.Version)
61 if err != nil {
62 return "", err
63 }
64 return filepath.Join(dir, encVer+"."+suffix), nil
65 }
66
67
68
69
70
71
72 func DownloadDir(ctx context.Context, m module.Version) (string, error) {
73 if gover.IsToolchain(m.Path) {
74 return "", ErrToolchain
75 }
76 if err := checkCacheDir(ctx); err != nil {
77 return "", err
78 }
79 enc, err := module.EscapePath(m.Path)
80 if err != nil {
81 return "", err
82 }
83 if !gover.ModIsValid(m.Path, m.Version) {
84 return "", fmt.Errorf("non-semver module version %q", m.Version)
85 }
86 if module.CanonicalVersion(m.Version) != m.Version {
87 return "", fmt.Errorf("non-canonical module version %q", m.Version)
88 }
89 encVer, err := module.EscapeVersion(m.Version)
90 if err != nil {
91 return "", err
92 }
93
94
95 dir := filepath.Join(cfg.GOMODCACHE, enc+"@"+encVer)
96 if fi, err := os.Stat(dir); os.IsNotExist(err) {
97 return dir, err
98 } else if err != nil {
99 return dir, &DownloadDirPartialError{dir, err}
100 } else if !fi.IsDir() {
101 return dir, &DownloadDirPartialError{dir, errors.New("not a directory")}
102 }
103
104
105
106 partialPath, err := CachePath(ctx, m, "partial")
107 if err != nil {
108 return dir, err
109 }
110 if _, err := os.Stat(partialPath); err == nil {
111 return dir, &DownloadDirPartialError{dir, errors.New("not completely extracted")}
112 } else if !os.IsNotExist(err) {
113 return dir, err
114 }
115
116
117
118
119 if m.Path == "golang.org/fips140" {
120 return dir, nil
121 }
122
123
124
125
126
127
128 ziphashPath, err := CachePath(ctx, m, "ziphash")
129 if err != nil {
130 return dir, err
131 }
132 if _, err := os.Stat(ziphashPath); os.IsNotExist(err) {
133 return dir, &DownloadDirPartialError{dir, errors.New("ziphash file is missing")}
134 } else if err != nil {
135 return dir, err
136 }
137 return dir, nil
138 }
139
140
141
142
143
144 type DownloadDirPartialError struct {
145 Dir string
146 Err error
147 }
148
149 func (e *DownloadDirPartialError) Error() string { return fmt.Sprintf("%s: %v", e.Dir, e.Err) }
150 func (e *DownloadDirPartialError) Is(err error) bool { return err == fs.ErrNotExist }
151
152
153
154 func lockVersion(ctx context.Context, mod module.Version) (unlock func(), err error) {
155 path, err := CachePath(ctx, mod, "lock")
156 if err != nil {
157 return nil, err
158 }
159 if err := os.MkdirAll(filepath.Dir(path), 0777); err != nil {
160 return nil, err
161 }
162 return lockedfile.MutexAt(path).Lock()
163 }
164
165
166
167
168
169 func SideLock(ctx context.Context) (unlock func(), err error) {
170 if err := checkCacheDir(ctx); err != nil {
171 return nil, err
172 }
173
174 path := filepath.Join(cfg.GOMODCACHE, "cache", "lock")
175 if err := os.MkdirAll(filepath.Dir(path), 0777); err != nil {
176 return nil, fmt.Errorf("failed to create cache directory: %w", err)
177 }
178
179 return lockedfile.MutexAt(path).Lock()
180 }
181
182
183
184
185
186
187 type cachingRepo struct {
188 path string
189 versionsCache par.ErrCache[string, *Versions]
190 statCache par.ErrCache[string, *RevInfo]
191 latestCache par.ErrCache[struct{}, *RevInfo]
192 gomodCache par.ErrCache[string, []byte]
193
194 once sync.Once
195 initRepo func(context.Context) (Repo, error)
196 r Repo
197 }
198
199 func newCachingRepo(ctx context.Context, path string, initRepo func(context.Context) (Repo, error)) *cachingRepo {
200 return &cachingRepo{
201 path: path,
202 initRepo: initRepo,
203 }
204 }
205
206 func (r *cachingRepo) repo(ctx context.Context) Repo {
207 r.once.Do(func() {
208 var err error
209 r.r, err = r.initRepo(ctx)
210 if err != nil {
211 r.r = errRepo{r.path, err}
212 }
213 })
214 return r.r
215 }
216
217 func (r *cachingRepo) CheckReuse(ctx context.Context, old *codehost.Origin) error {
218 return r.repo(ctx).CheckReuse(ctx, old)
219 }
220
221 func (r *cachingRepo) ModulePath() string {
222 return r.path
223 }
224
225 func (r *cachingRepo) Versions(ctx context.Context, prefix string) (*Versions, error) {
226 v, err := r.versionsCache.Do(prefix, func() (*Versions, error) {
227 return r.repo(ctx).Versions(ctx, prefix)
228 })
229
230 if err != nil {
231 return nil, err
232 }
233 return &Versions{
234 Origin: v.Origin,
235 List: append([]string(nil), v.List...),
236 }, nil
237 }
238
239 type cachedInfo struct {
240 info *RevInfo
241 err error
242 }
243
244 func (r *cachingRepo) Stat(ctx context.Context, rev string) (*RevInfo, error) {
245 if gover.IsToolchain(r.path) {
246
247 return r.repo(ctx).Stat(ctx, rev)
248 }
249 info, err := r.statCache.Do(rev, func() (*RevInfo, error) {
250 file, info, err := readDiskStat(ctx, r.path, rev)
251 if err == nil {
252 return info, err
253 }
254
255 info, err = r.repo(ctx).Stat(ctx, rev)
256 if err == nil {
257
258
259 if info.Version != rev {
260 file, _ = CachePath(ctx, module.Version{Path: r.path, Version: info.Version}, "info")
261 r.statCache.Do(info.Version, func() (*RevInfo, error) {
262 return info, nil
263 })
264 }
265
266 if err := writeDiskStat(ctx, file, info); err != nil {
267 fmt.Fprintf(os.Stderr, "go: writing stat cache: %v\n", err)
268 }
269 }
270 return info, err
271 })
272 if info != nil {
273 copy := *info
274 info = ©
275 }
276 return info, err
277 }
278
279 func (r *cachingRepo) Latest(ctx context.Context) (*RevInfo, error) {
280 if gover.IsToolchain(r.path) {
281
282 return r.repo(ctx).Latest(ctx)
283 }
284 info, err := r.latestCache.Do(struct{}{}, func() (*RevInfo, error) {
285 info, err := r.repo(ctx).Latest(ctx)
286
287
288 if err == nil {
289 r.statCache.Do(info.Version, func() (*RevInfo, error) {
290 return info, nil
291 })
292 if file, _, err := readDiskStat(ctx, r.path, info.Version); err != nil {
293 writeDiskStat(ctx, file, info)
294 }
295 }
296
297 return info, err
298 })
299 if info != nil {
300 copy := *info
301 info = ©
302 }
303 return info, err
304 }
305
306 func (r *cachingRepo) GoMod(ctx context.Context, version string) ([]byte, error) {
307 if gover.IsToolchain(r.path) {
308
309 return r.repo(ctx).GoMod(ctx, version)
310 }
311 text, err := r.gomodCache.Do(version, func() ([]byte, error) {
312 file, text, err := readDiskGoMod(ctx, r.path, version)
313 if err == nil {
314
315 return text, nil
316 }
317
318 text, err = r.repo(ctx).GoMod(ctx, version)
319 if err == nil {
320 if err := checkGoMod(r.path, version, text); err != nil {
321 return text, err
322 }
323 if err := writeDiskGoMod(ctx, file, text); err != nil {
324 fmt.Fprintf(os.Stderr, "go: writing go.mod cache: %v\n", err)
325 }
326 }
327 return text, err
328 })
329 if err != nil {
330 return nil, err
331 }
332 return append([]byte(nil), text...), nil
333 }
334
335 func (r *cachingRepo) Zip(ctx context.Context, dst io.Writer, version string) error {
336 if gover.IsToolchain(r.path) {
337 return ErrToolchain
338 }
339 return r.repo(ctx).Zip(ctx, dst, version)
340 }
341
342
343
344 func InfoFile(ctx context.Context, path, version string) (*RevInfo, string, error) {
345 if !gover.ModIsValid(path, version) {
346 return nil, "", fmt.Errorf("invalid version %q", version)
347 }
348
349 if file, info, err := readDiskStat(ctx, path, version); err == nil {
350 return info, file, nil
351 }
352
353 var info *RevInfo
354 var err2info map[error]*RevInfo
355 err := TryProxies(func(proxy string) error {
356 i, err := Lookup(ctx, proxy, path).Stat(ctx, version)
357 if err == nil {
358 info = i
359 } else {
360 if err2info == nil {
361 err2info = make(map[error]*RevInfo)
362 }
363 err2info[err] = info
364 }
365 return err
366 })
367 if err != nil {
368 return err2info[err], "", err
369 }
370
371
372 file, err := CachePath(ctx, module.Version{Path: path, Version: version}, "info")
373 if err != nil {
374 return nil, "", err
375 }
376 return info, file, nil
377 }
378
379
380
381
382 func GoMod(ctx context.Context, path, rev string) ([]byte, error) {
383
384
385 if !gover.ModIsValid(path, rev) {
386 if _, info, err := readDiskStat(ctx, path, rev); err == nil {
387 rev = info.Version
388 } else {
389 if errors.Is(err, statCacheErr) {
390 return nil, err
391 }
392 err := TryProxies(func(proxy string) error {
393 info, err := Lookup(ctx, proxy, path).Stat(ctx, rev)
394 if err == nil {
395 rev = info.Version
396 }
397 return err
398 })
399 if err != nil {
400 return nil, err
401 }
402 }
403 }
404
405 _, data, err := readDiskGoMod(ctx, path, rev)
406 if err == nil {
407 return data, nil
408 }
409
410 err = TryProxies(func(proxy string) (err error) {
411 data, err = Lookup(ctx, proxy, path).GoMod(ctx, rev)
412 return err
413 })
414 return data, err
415 }
416
417
418
419 func GoModFile(ctx context.Context, path, version string) (string, error) {
420 if !gover.ModIsValid(path, version) {
421 return "", fmt.Errorf("invalid version %q", version)
422 }
423 if _, err := GoMod(ctx, path, version); err != nil {
424 return "", err
425 }
426
427 file, err := CachePath(ctx, module.Version{Path: path, Version: version}, "mod")
428 if err != nil {
429 return "", err
430 }
431 return file, nil
432 }
433
434
435
436 func GoModSum(ctx context.Context, path, version string) (string, error) {
437 if !gover.ModIsValid(path, version) {
438 return "", fmt.Errorf("invalid version %q", version)
439 }
440 data, err := GoMod(ctx, path, version)
441 if err != nil {
442 return "", err
443 }
444 sum, err := goModSum(data)
445 if err != nil {
446 return "", err
447 }
448 return sum, nil
449 }
450
451 var errNotCached = fmt.Errorf("not in cache")
452
453
454
455
456
457 func readDiskStat(ctx context.Context, path, rev string) (file string, info *RevInfo, err error) {
458 if gover.IsToolchain(path) {
459 return "", nil, errNotCached
460 }
461 file, data, err := readDiskCache(ctx, path, rev, "info")
462 if err != nil {
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482 if cfg.GOPROXY == "off" {
483 if file, info, err := readDiskStatByHash(ctx, path, rev); err == nil {
484 return file, info, nil
485 }
486 }
487 return file, nil, err
488 }
489 info = new(RevInfo)
490 if err := json.Unmarshal(data, info); err != nil {
491 return file, nil, errNotCached
492 }
493
494
495
496 data2, err := json.Marshal(info)
497 if err == nil && !bytes.Equal(data2, data) {
498 writeDiskCache(ctx, file, data)
499 }
500 return file, info, nil
501 }
502
503
504
505
506
507
508
509
510
511
512 func readDiskStatByHash(ctx context.Context, path, rev string) (file string, info *RevInfo, err error) {
513 if gover.IsToolchain(path) {
514 return "", nil, errNotCached
515 }
516 if cfg.GOMODCACHE == "" {
517
518 return "", nil, errNotCached
519 }
520
521 if !codehost.AllHex(rev) || len(rev) < 12 {
522 return "", nil, errNotCached
523 }
524 rev = rev[:12]
525 cdir, err := cacheDir(ctx, path)
526 if err != nil {
527 return "", nil, errNotCached
528 }
529 dir, err := os.Open(cdir)
530 if err != nil {
531 return "", nil, errNotCached
532 }
533 names, err := dir.Readdirnames(-1)
534 dir.Close()
535 if err != nil {
536 return "", nil, errNotCached
537 }
538
539
540
541
542 var maxVersion string
543 suffix := "-" + rev + ".info"
544 err = errNotCached
545 for _, name := range names {
546 if strings.HasSuffix(name, suffix) {
547 v := strings.TrimSuffix(name, ".info")
548 if module.IsPseudoVersion(v) && semver.Compare(v, maxVersion) > 0 {
549 maxVersion = v
550 file, info, err = readDiskStat(ctx, path, strings.TrimSuffix(name, ".info"))
551 }
552 }
553 }
554 return file, info, err
555 }
556
557
558
559
560
561
562 var oldVgoPrefix = []byte("//vgo 0.0.")
563
564
565
566
567
568 func readDiskGoMod(ctx context.Context, path, rev string) (file string, data []byte, err error) {
569 if gover.IsToolchain(path) {
570 return "", nil, errNotCached
571 }
572 file, data, err = readDiskCache(ctx, path, rev, "mod")
573
574
575 if bytes.HasPrefix(data, oldVgoPrefix) {
576 err = errNotCached
577 data = nil
578 }
579
580 if err == nil {
581 if err := checkGoMod(path, rev, data); err != nil {
582 return "", nil, err
583 }
584 }
585
586 return file, data, err
587 }
588
589
590
591
592
593
594 func readDiskCache(ctx context.Context, path, rev, suffix string) (file string, data []byte, err error) {
595 if gover.IsToolchain(path) {
596 return "", nil, errNotCached
597 }
598 file, err = CachePath(ctx, module.Version{Path: path, Version: rev}, suffix)
599 if err != nil {
600 return "", nil, errNotCached
601 }
602 data, err = robustio.ReadFile(file)
603 if err != nil {
604 return file, nil, errNotCached
605 }
606 return file, data, nil
607 }
608
609
610
611 func writeDiskStat(ctx context.Context, file string, info *RevInfo) error {
612 if file == "" {
613 return nil
614 }
615
616 if info.Origin != nil {
617
618
619
620 clean := *info
621 info = &clean
622 o := *info.Origin
623 info.Origin = &o
624
625
626
627 o.TagSum = ""
628 o.TagPrefix = ""
629
630 if module.IsPseudoVersion(info.Version) {
631 o.Ref = ""
632 }
633 }
634
635 js, err := json.Marshal(info)
636 if err != nil {
637 return err
638 }
639 return writeDiskCache(ctx, file, js)
640 }
641
642
643
644 func writeDiskGoMod(ctx context.Context, file string, text []byte) error {
645 return writeDiskCache(ctx, file, text)
646 }
647
648
649
650 func writeDiskCache(ctx context.Context, file string, data []byte) error {
651 if file == "" {
652 return nil
653 }
654
655 if err := os.MkdirAll(filepath.Dir(file), 0777); err != nil {
656 return err
657 }
658
659
660
661 f, err := tempFile(ctx, filepath.Dir(file), filepath.Base(file), 0666)
662 if err != nil {
663 return err
664 }
665 defer func() {
666
667
668
669 if err != nil {
670 f.Close()
671 os.Remove(f.Name())
672 }
673 }()
674
675 if _, err := f.Write(data); err != nil {
676 return err
677 }
678 if err := f.Close(); err != nil {
679 return err
680 }
681 if err := robustio.Rename(f.Name(), file); err != nil {
682 return err
683 }
684
685 if strings.HasSuffix(file, ".mod") {
686 rewriteVersionList(ctx, filepath.Dir(file))
687 }
688 return nil
689 }
690
691
692 func tempFile(ctx context.Context, dir, prefix string, perm fs.FileMode) (f *os.File, err error) {
693 for i := 0; i < 10000; i++ {
694 name := filepath.Join(dir, prefix+strconv.Itoa(rand.Intn(1000000000))+".tmp")
695 f, err = os.OpenFile(name, os.O_RDWR|os.O_CREATE|os.O_EXCL, perm)
696 if os.IsExist(err) {
697 if ctx.Err() != nil {
698 return nil, ctx.Err()
699 }
700 continue
701 }
702 break
703 }
704 return
705 }
706
707
708
709 func rewriteVersionList(ctx context.Context, dir string) (err error) {
710 if filepath.Base(dir) != "@v" {
711 base.Fatalf("go: internal error: misuse of rewriteVersionList")
712 }
713
714 listFile := filepath.Join(dir, "list")
715
716
717
718
719
720
721
722
723
724
725 f, err := lockedfile.Edit(listFile)
726 if err != nil {
727 return err
728 }
729 defer func() {
730 if cerr := f.Close(); cerr != nil && err == nil {
731 err = cerr
732 }
733 }()
734 infos, err := os.ReadDir(dir)
735 if err != nil {
736 return err
737 }
738 var list []string
739 for _, info := range infos {
740
741
742
743
744
745
746 name := info.Name()
747 if v, found := strings.CutSuffix(name, ".mod"); found {
748 if v != "" && module.CanonicalVersion(v) == v {
749 list = append(list, v)
750 }
751 }
752 }
753 semver.Sort(list)
754
755 var buf bytes.Buffer
756 for _, v := range list {
757 buf.WriteString(v)
758 buf.WriteString("\n")
759 }
760 if fi, err := f.Stat(); err == nil && int(fi.Size()) == buf.Len() {
761 old := make([]byte, buf.Len()+1)
762 if n, err := f.ReadAt(old, 0); err == io.EOF && n == buf.Len() && bytes.Equal(buf.Bytes(), old) {
763 return nil
764 }
765 }
766
767
768 if err := f.Truncate(0); err != nil {
769 return err
770 }
771
772 if err := f.Truncate(int64(buf.Len())); err != nil {
773 return err
774 }
775
776
777 if _, err := f.Write(buf.Bytes()); err != nil {
778 f.Truncate(0)
779 return err
780 }
781
782 return nil
783 }
784
785 var (
786 statCacheOnce sync.Once
787 statCacheErr error
788
789 counterErrorsGOMODCACHEEntryRelative = counter.New("go/errors:gomodcache-entry-relative")
790 )
791
792
793
794 func checkCacheDir(ctx context.Context) error {
795 if cfg.GOMODCACHE == "" {
796
797
798 return fmt.Errorf("module cache not found: neither GOMODCACHE nor GOPATH is set")
799 }
800 if !filepath.IsAbs(cfg.GOMODCACHE) {
801 counterErrorsGOMODCACHEEntryRelative.Inc()
802 return fmt.Errorf("GOMODCACHE entry is relative; must be absolute path: %q.\n", cfg.GOMODCACHE)
803 }
804
805
806
807 statCacheOnce.Do(func() {
808 fi, err := os.Stat(cfg.GOMODCACHE)
809 if err != nil {
810 if !os.IsNotExist(err) {
811 statCacheErr = fmt.Errorf("could not create module cache: %w", err)
812 return
813 }
814 if err := os.MkdirAll(cfg.GOMODCACHE, 0777); err != nil {
815 statCacheErr = fmt.Errorf("could not create module cache: %w", err)
816 return
817 }
818 return
819 }
820 if !fi.IsDir() {
821 statCacheErr = fmt.Errorf("could not create module cache: %q is not a directory", cfg.GOMODCACHE)
822 return
823 }
824 })
825 return statCacheErr
826 }
827
View as plain text