forked from gnolang/gno
-
Notifications
You must be signed in to change notification settings - Fork 0
/
memfile.go
112 lines (95 loc) · 3.1 KB
/
memfile.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
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
package gnovm
import (
"fmt"
"regexp"
"sort"
"strings"
)
type MemFile struct {
Name string `json:"name" yaml:"name"`
Body string `json:"body" yaml:"body"`
}
// MemPackage represents the information and files of a package which will be
// stored in memory. It will generally be initialized by package gnolang's
// ReadMemPackage.
//
// NOTE: in the future, a MemPackage may represent
// updates/additional-files for an existing package.
type MemPackage struct {
Name string `json:"name" yaml:"name"` // package name as declared by `package`
Path string `json:"path" yaml:"path"` // import path
Files []*MemFile `json:"files" yaml:"files"`
}
func (mempkg *MemPackage) GetFile(name string) *MemFile {
for _, memFile := range mempkg.Files {
if memFile.Name == name {
return memFile
}
}
return nil
}
func (mempkg *MemPackage) IsEmpty() bool {
return len(mempkg.Files) == 0
}
const pathLengthLimit = 256
var (
rePkgName = regexp.MustCompile(`^[a-z][a-z0-9_]*$`)
rePkgOrRlmPath = regexp.MustCompile(`^([a-zA-Z0-9-]+\.)*[a-zA-Z0-9-]+\.[a-zA-Z]{2,}\/(?:p|r)(?:\/_?[a-z]+[a-z0-9_]*)+$`)
reFileName = regexp.MustCompile(`^([a-zA-Z0-9_]*\.[a-z0-9_\.]*|LICENSE|README)$`)
)
// path must not contain any dots after the first domain component.
// file names must contain dots.
// NOTE: this is to prevent conflicts with nested paths.
func (mempkg *MemPackage) Validate() error {
// add assertion that MemPkg contains at least 1 file
if len(mempkg.Files) <= 0 {
return fmt.Errorf("no files found within package %q", mempkg.Name)
}
if len(mempkg.Path) > pathLengthLimit {
return fmt.Errorf("path length %d exceeds limit %d", len(mempkg.Path), pathLengthLimit)
}
if !rePkgName.MatchString(mempkg.Name) {
return fmt.Errorf("invalid package name %q, failed to match %q", mempkg.Name, rePkgName)
}
if !rePkgOrRlmPath.MatchString(mempkg.Path) {
return fmt.Errorf("invalid package/realm path %q, failed to match %q", mempkg.Path, rePkgOrRlmPath)
}
// enforce sorting files based on Go conventions for predictability
sorted := sort.SliceIsSorted(
mempkg.Files,
func(i, j int) bool {
return mempkg.Files[i].Name < mempkg.Files[j].Name
},
)
if !sorted {
return fmt.Errorf("mempackage %q has unsorted files", mempkg.Path)
}
var prev string
for i, file := range mempkg.Files {
if !reFileName.MatchString(file.Name) {
return fmt.Errorf("invalid file name %q, failed to match %q", file.Name, reFileName)
}
if i > 0 && prev == file.Name {
return fmt.Errorf("duplicate file name %q", file.Name)
}
prev = file.Name
}
return nil
}
const licenseName = "LICENSE"
// Splits a path into the dirpath and filename.
func SplitFilepath(filepath string) (dirpath string, filename string) {
parts := strings.Split(filepath, "/")
if len(parts) == 1 {
return parts[0], ""
}
switch last := parts[len(parts)-1]; {
case strings.Contains(last, "."):
return strings.Join(parts[:len(parts)-1], "/"), last
case last == "":
return strings.Join(parts[:len(parts)-1], "/"), ""
case last == licenseName:
return strings.Join(parts[:len(parts)-1], "/"), licenseName
}
return strings.Join(parts, "/"), ""
}