Skip to content

Commit 2caea05

Browse files
Add mindev ruletype init to kick off a rule type (#5228)
* Add `mindev ruletype init` to kick off a rule type This helps folks set up the basic skeleton for ruletype writing. Signed-off-by: Juan Antonio Osorio <[email protected]> * Apply suggestions from code review Co-authored-by: Evan Anderson <[email protected]> * More fixes Signed-off-by: Juan Antonio Osorio <[email protected]> * address lint issue Signed-off-by: Juan Antonio Osorio <[email protected]> --------- Signed-off-by: Juan Antonio Osorio <[email protected]> Co-authored-by: Evan Anderson <[email protected]>
1 parent d5c676c commit 2caea05

File tree

2 files changed

+206
-0
lines changed

2 files changed

+206
-0
lines changed

cmd/dev/app/rule_type/init.go

+205
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,205 @@
1+
// SPDX-FileCopyrightText: Copyright 2023 The Minder Authors
2+
// SPDX-License-Identifier: Apache-2.0
3+
4+
package rule_type
5+
6+
import (
7+
"bytes"
8+
"errors"
9+
"fmt"
10+
"os"
11+
"path/filepath"
12+
"regexp"
13+
"text/template"
14+
15+
"github.com/spf13/cobra"
16+
)
17+
18+
// CmdInit is the command for initializing a rule type definition
19+
func CmdInit() *cobra.Command {
20+
initCmd := &cobra.Command{
21+
Use: "init",
22+
Short: "initialize a rule type definition",
23+
Long: `The 'ruletype init' subcommand allows you to initialize a rule type definition
24+
25+
The first positional argument is the directory to initialize the rule type in.
26+
The rule type will be initialized in the current directory if no directory is provided.
27+
`,
28+
RunE: initCmdRun,
29+
SilenceUsage: true,
30+
}
31+
32+
initCmd.Flags().StringP("name", "n", "", "name of the rule type")
33+
initCmd.Flags().BoolP("skip-tests", "s", false, "skip creating test files")
34+
35+
if err := initCmd.MarkFlagRequired("name"); err != nil {
36+
fmt.Fprintf(os.Stderr, "Error marking flag as required: %s\n", err)
37+
os.Exit(1)
38+
}
39+
40+
return initCmd
41+
}
42+
43+
func initCmdRun(cmd *cobra.Command, args []string) error {
44+
name := cmd.Flag("name").Value.String()
45+
skipTests := cmd.Flag("skip-tests").Value.String() == "true"
46+
dir := "."
47+
if len(args) > 0 {
48+
dir = args[0]
49+
}
50+
51+
if err := validateRuleTypeName(name); err != nil {
52+
return err
53+
}
54+
55+
ruleTypeFileName := filepath.Join(dir, name+".yaml")
56+
ruleTypeTestFileName := filepath.Join(dir, name+".test.yaml")
57+
ruleTypeTestDataDirName := filepath.Join(dir, name+".testdata")
58+
59+
if err := assertFilesDontExist(
60+
ruleTypeFileName, ruleTypeTestFileName, ruleTypeTestDataDirName); err != nil {
61+
return err
62+
}
63+
64+
// Create rule type file
65+
if err := createRuleTypeFile(ruleTypeFileName, name); err != nil {
66+
return err
67+
}
68+
cmd.Printf("Created rule type file: %s\n", ruleTypeFileName)
69+
70+
if !skipTests {
71+
// Create rule type test file
72+
if err := createRuleTypeTestFile(ruleTypeTestFileName, name); err != nil {
73+
return err
74+
}
75+
cmd.Printf("Created rule type test file: %s\n", ruleTypeTestFileName)
76+
77+
// Create rule type test data directory
78+
if err := createRuleTypeTestDataDir(ruleTypeTestDataDirName); err != nil {
79+
return err
80+
}
81+
cmd.Printf("Created rule type test data directory: %s\n", ruleTypeTestDataDirName)
82+
}
83+
84+
return nil
85+
}
86+
87+
func validateRuleTypeName(name string) error {
88+
if name == "" {
89+
return errors.New("name cannot be empty")
90+
}
91+
92+
validName := regexp.MustCompile(`^[A-Za-z][-/[:word:]]*$`)
93+
94+
// regexp to validate name
95+
if !validName.MatchString(name) {
96+
return errors.New("name must only contain alphanumeric characters and underscores")
97+
}
98+
99+
return nil
100+
}
101+
102+
func assertFilesDontExist(files ...string) error {
103+
for _, file := range files {
104+
_, err := os.Stat(file)
105+
if err == nil {
106+
return fmt.Errorf("file %s already exists", file)
107+
}
108+
109+
if errors.Is(err, os.ErrPermission) {
110+
return fmt.Errorf("permission denied for file %s", file)
111+
}
112+
}
113+
114+
return nil
115+
}
116+
117+
func createRuleTypeFile(fileName, name string) error {
118+
return createFileWithContent(fileName, renderwithRuleTypeName(`---
119+
version: v1
120+
release_phase: alpha
121+
type: rule-type
122+
name: {{ .RuleName }}
123+
display_name: # Display name for the rule type
124+
short_failure_message: # Short message to display when the rule fails
125+
severity:
126+
value: medium
127+
context: {}
128+
description: | # Description of the rule type
129+
guidance: | # Guidance for the rule type. This helps users understand how to fix the issue.
130+
def:
131+
in_entity: repository # The entity type the rule applies to
132+
rule_schema: {}
133+
ingest:
134+
type: git
135+
git:
136+
eval:
137+
type: rego
138+
rego:
139+
type: deny-by-default
140+
def: |
141+
package minder
142+
143+
import rego.v1
144+
145+
default allow := false
146+
147+
allow if {
148+
true
149+
}
150+
151+
message := "This is a test message"
152+
`, name))
153+
}
154+
155+
func createRuleTypeTestFile(fileName, name string) error {
156+
return createFileWithContent(fileName, renderwithRuleTypeName(`---
157+
tests:
158+
- name: "TEST NAME GOES HERE""
159+
def: {}
160+
params: {}
161+
expect: "pass"
162+
entity: &test-repo
163+
type: repository
164+
entity:
165+
owner: "coolhead"
166+
name: "haze-wave"
167+
# When testing a rule, additional content can be supplied
168+
# from files in the "{{ .RuleName }}.testdata" directory.
169+
# File paths below are relative to this directory.
170+
# http:
171+
# # Input from the "http" ingest type.
172+
# body_file: HTTP_BODY_FILE
173+
# git:
174+
# # Input from the "git" ingest type. Base paths contain
175+
# # directory contents, but do not actually need to be a
176+
# # git repository.
177+
# repo_base: REPO_BASE_PATH
178+
`, name))
179+
}
180+
181+
func createRuleTypeTestDataDir(dirName string) error {
182+
if err := os.Mkdir(dirName, 0750); err != nil {
183+
return fmt.Errorf("error creating directory %s: %w", dirName, err)
184+
}
185+
186+
return nil
187+
}
188+
189+
func createFileWithContent(fileName, content string) error {
190+
return os.WriteFile(fileName, []byte(content), 0600)
191+
}
192+
193+
func renderwithRuleTypeName(templ, name string) string {
194+
tmpl, err := template.New("ruletype").Parse(templ)
195+
if err != nil {
196+
panic(err)
197+
}
198+
199+
var buf bytes.Buffer
200+
if err := tmpl.Execute(&buf, struct{ RuleName string }{name}); err != nil {
201+
panic(err)
202+
}
203+
204+
return buf.String()
205+
}

cmd/dev/app/rule_type/ruletype.go

+1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ func CmdRuleType() *cobra.Command {
1616
rtCmd.AddCommand(CmdTest())
1717
rtCmd.AddCommand(CmdLint())
1818
rtCmd.AddCommand(CmdValidateUpdate())
19+
rtCmd.AddCommand(CmdInit())
1920

2021
return rtCmd
2122
}

0 commit comments

Comments
 (0)