forked from hanksudo/prototool
-
Notifications
You must be signed in to change notification settings - Fork 0
/
check_file_options_equal.go
120 lines (105 loc) · 4.55 KB
/
check_file_options_equal.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
113
114
115
116
117
118
119
120
// Copyright (c) 2018 Uber Technologies, Inc.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
package lint
import (
"fmt"
"github.com/emicklei/proto"
"github.com/uber/prototool/internal/protostrs"
"github.com/uber/prototool/internal/text"
)
var fileOptionsEqualGoPackagePbSuffixLinter = NewLinter(
"FILE_OPTIONS_EQUAL_GO_PACKAGE_PB_SUFFIX",
`Verifies that the file option "go_package" is equal to $(basename PACKAGE)pb.`,
newCheckFileOptionsEqual("go_package", func(_ *proto.Proto, pkg *proto.Package) string {
return protostrs.GoPackage(pkg.Name)
}),
)
var fileOptionsEqualJavaMultipleFilesTrueLinter = NewLinter(
"FILE_OPTIONS_EQUAL_JAVA_MULTIPLE_FILES_TRUE",
`Verifies that the file option "java_multiple_files" is equal to true.`,
newCheckFileOptionsEqual("java_multiple_files", func(*proto.Proto, *proto.Package) string {
return "true"
}),
)
var fileOptionsEqualJavaOuterClassnameProtoSuffixLinter = NewLinter(
"FILE_OPTIONS_EQUAL_JAVA_OUTER_CLASSNAME_PROTO_SUFFIX",
`Verifies that the file option "java_outer_classname" is equal to $(upperCamelCase $(basename FILE))Proto.`,
newCheckFileOptionsEqual("java_outer_classname", func(descriptor *proto.Proto, _ *proto.Package) string {
return protostrs.JavaOuterClassname(descriptor.Filename)
}),
)
var fileOptionsEqualJavaPackageComPrefixLinter = NewLinter(
"FILE_OPTIONS_EQUAL_JAVA_PACKAGE_COM_PREFIX",
`Verifies that the file option "java_package" is equal to com.PACKAGE.`,
newCheckFileOptionsEqual("java_package", func(_ *proto.Proto, pkg *proto.Package) string {
return protostrs.JavaPackage(pkg.Name)
}),
)
func newCheckFileOptionsEqual(fileOption string, expectedValueFunc func(*proto.Proto, *proto.Package) string) func(func(*text.Failure), string, []*proto.Proto) error {
return func(add func(*text.Failure), dirPath string, descriptors []*proto.Proto) error {
return runVisitor(&fileOptionsEqualVisitor{
baseAddVisitor: newBaseAddVisitor(add),
fileOption: fileOption,
expectedValueFunc: expectedValueFunc,
}, descriptors)
}
}
type fileOptionsEqualVisitor struct {
baseAddVisitor
fileOption string
expectedValueFunc func(*proto.Proto, *proto.Package) string
descriptor *proto.Proto
pkg *proto.Package
option *proto.Option
}
func (v *fileOptionsEqualVisitor) OnStart(descriptor *proto.Proto) error {
v.descriptor = descriptor
v.pkg = nil
v.option = nil
return nil
}
func (v *fileOptionsEqualVisitor) VisitPackage(element *proto.Package) {
v.pkg = element
}
func (v *fileOptionsEqualVisitor) VisitOption(element *proto.Option) {
// TODO: not validating this is a file option, or are we since we're not recursing on other elements?
if element.Name == v.fileOption {
v.option = element
}
}
func (v *fileOptionsEqualVisitor) Finally() error {
if v.descriptor == nil || v.pkg == nil || v.option == nil {
// do not do anything, other linters should verify that the file option exists
// this makes it possible to be optional if a required file option linter is suppressed
// TODO make sure this is consistent across all linters
return nil
}
if v.descriptor.Filename == "" {
// if this isn't set, we made a mistake setting this up, return a system error
return fmt.Errorf("expected filename to be set for descriptor %v in checkFileOptionsEqual linter", v.descriptor)
}
// TODO: handle AggregatedConstants
value := v.option.Constant.Source
expectedValue := v.expectedValueFunc(v.descriptor, v.pkg)
if expectedValue != value {
v.AddFailuref(v.option.Position, "Expected %q for option %q but was %q.", expectedValue, v.option.Name, value)
}
return nil
}