forked from sourcegraph/sourcegraph-public-snapshot
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathinventory.go
88 lines (80 loc) · 3.18 KB
/
inventory.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
package backend
import (
"context"
"encoding/json"
"fmt"
"io"
"os"
"strconv"
"github.com/inconshreveable/log15"
"github.com/pkg/errors"
"github.com/sourcegraph/sourcegraph/cmd/frontend/internal/inventory"
"github.com/sourcegraph/sourcegraph/internal/api"
"github.com/sourcegraph/sourcegraph/internal/env"
"github.com/sourcegraph/sourcegraph/internal/gitserver"
"github.com/sourcegraph/sourcegraph/internal/rcache"
"github.com/sourcegraph/sourcegraph/internal/vcs/git"
)
// Feature flag for enhanced (but much slower) language detection that uses file contents, not just
// filenames. Enabled by default.
var useEnhancedLanguageDetection, _ = strconv.ParseBool(env.Get("USE_ENHANCED_LANGUAGE_DETECTION", "true", "Enable more accurate but slower language detection that uses file contents"))
var inventoryCache = rcache.New(fmt.Sprintf("inv:v2:enhanced_%v", useEnhancedLanguageDetection))
// InventoryContext returns the inventory context for computing the inventory for the repository at
// the given commit.
func InventoryContext(repo gitserver.Repo, commitID api.CommitID, forceEnhancedLanguageDetection bool) (inventory.Context, error) {
if !git.IsAbsoluteRevision(string(commitID)) {
return inventory.Context{}, errors.Errorf("refusing to compute inventory for non-absolute commit ID %q", commitID)
}
cacheKey := func(e os.FileInfo) string {
info, ok := e.Sys().(git.ObjectInfo)
if !ok {
return "" // not cacheable
}
return info.OID().String()
}
invCtx := inventory.Context{
ReadTree: func(ctx context.Context, path string) ([]os.FileInfo, error) {
// TODO: As a perf optimization, we could read multiple levels of the Git tree at once
// to avoid sequential tree traversal calls.
return git.ReadDir(ctx, repo, commitID, path, false)
},
NewFileReader: func(ctx context.Context, path string) (io.ReadCloser, error) {
return git.NewFileReader(ctx, repo, commitID, path)
},
CacheGet: func(e os.FileInfo) (inventory.Inventory, bool) {
cacheKey := cacheKey(e)
if cacheKey == "" {
return inventory.Inventory{}, false // not cacheable
}
if b, ok := inventoryCache.Get(cacheKey); ok {
var inv inventory.Inventory
if err := json.Unmarshal(b, &inv); err != nil {
log15.Warn("Failed to unmarshal cached JSON inventory.", "repo", repo.Name, "commitID", commitID, "path", e.Name(), "err", err)
return inventory.Inventory{}, false
}
return inv, true
}
return inventory.Inventory{}, false
},
CacheSet: func(e os.FileInfo, inv inventory.Inventory) {
cacheKey := cacheKey(e)
if cacheKey == "" {
return // not cacheable
}
b, err := json.Marshal(&inv)
if err != nil {
log15.Warn("Failed to marshal JSON inventory for cache.", "repo", repo.Name, "commitID", commitID, "path", e.Name(), "err", err)
return
}
inventoryCache.Set(cacheKey, b)
},
}
if !useEnhancedLanguageDetection && !forceEnhancedLanguageDetection {
// If USE_ENHANCED_LANGUAGE_DETECTION is disabled, do not read file contents to determine
// the language. This means we won't calculate the number of lines per language.
invCtx.NewFileReader = func(ctx context.Context, path string) (io.ReadCloser, error) {
return nil, nil
}
}
return invCtx, nil
}