Skip to content

Commit cf3be10

Browse files
better script options
1 parent b27199e commit cf3be10

File tree

8 files changed

+78
-41
lines changed

8 files changed

+78
-41
lines changed

Readme.md

+7-6
Original file line numberDiff line numberDiff line change
@@ -270,16 +270,16 @@ The check produces an offender if the script produced output on stdout or stderr
270270

271271
- `File` : string, the full path of the file or directory
272272
- `Script` : string, the full path of the script
273-
- `ScriptOptions` : string array, (optional) the first element allows to define a pattern containing wildcards like `?`, `*`, and `**` that is applied to filenames if present it will only check files that match the pattern, this is mostly useful when running the script on a directory. The second element allows passing arguments to the script.
273+
- `ScriptOptions` : string array, (optional) the first element allows to define a pattern containing wildcards like `?`, `*`, and `**` that is applied to filenames if present it will only check files that match the pattern, this is mostly useful when running the script on a directory. Arguments can be passed to the script using the second and following elements.
274274
- `File` : string, the full path of the file, if the path points to a directory the script is run for every file in the directory and subdirectories
275275

276276
- `Desc` : string, (optional) is a descriptive string that will be attached to failed check
277277
- `InformationalOnly` : bool, (optional) the result of the check will be Informational only (default: false)
278278

279-
If the `--` is present it indicates that the next argument is from the `ScriptOptions[1]`. The script is run with the following arguments:
279+
If the `--` is present it indicates that the next argument is from the `ScriptOptions[1..N]`. The script is run with the following arguments:
280280

281281
```
282-
<tmp filename> <original filename> <uid> <gid> <mode in octal> <selinux label or "-" for no label> [--] [script options argument]
282+
<tmp filename> <original filename> <uid> <gid> <mode in octal> <selinux label or "-" for no label> [--] [script argument 1] ... [script argument N]
283283
```
284284

285285
Example:
@@ -345,13 +345,13 @@ as arguments. If the script prints output the check will be marked as failed.
345345

346346
- `File` : string, the full path of the file
347347
- `Script`: string, path to the script
348-
- `ScriptOptions`: string, argument passed to the script
348+
- `ScriptOptions`: string array, (optional) arguments passed to the script
349349
- `OldFilePath`: string, filename (absolute or relative) to use to store old file
350350
- `InformationalOnly` : bool, (optional) the result of the check will be Informational only (default: false)
351351

352352
Script runs as:
353353
```sh
354-
script.sh <OrigFilename> <oldFile> <newFile> -- <argument>
354+
script.sh <OrigFilename> <oldFile> <newFile> [--] [argument 1] .. [argument N]
355355
```
356356

357357
Example:
@@ -465,14 +465,15 @@ this statement, the example below is named LastLine.
465465

466466
- `File` : string, the full path of the file
467467
- `Script` :string, the full path of the script
468+
- `ScriptOptions` : string array (optionl), arguments to pass to the script
468469
- `Name` : string, (optional) the key name
469470

470471
- `Desc` : string, (optional) description
471472

472473
The script is run with the following arguments:
473474

474475
```
475-
<tmp filename> <original filename> <uid> <gid> <mode in octal> <selinux label or "-" for no label>
476+
<tmp filename> <original filename> <uid> <gid> <mode in octal> <selinux label or "-" for no label> [--] [script argument 1] ... [script argument N]
476477
```
477478

478479
Example:

pkg/analyzer/dataextract/dataextract.go

+17-11
Original file line numberDiff line numberDiff line change
@@ -34,12 +34,13 @@ import (
3434
)
3535

3636
type dataType struct {
37-
File string
38-
Script string
39-
RegEx string
40-
Json string
41-
Desc string
42-
Name string // the name can be set directly otherwise the key will be used
37+
File string
38+
Script string
39+
ScriptOptions []string // options for script execution
40+
RegEx string
41+
Json string
42+
Desc string
43+
Name string // the name can be set directly otherwise the key will be used
4344
}
4445

4546
type dataExtractType struct {
@@ -152,7 +153,7 @@ func (state *dataExtractType) CheckFile(fi *fsparser.FileInfo, filepath string)
152153
}
153154

154155
if item.Script != "" {
155-
out, err := runScriptOnFile(state.a, item.Script, fi, fn)
156+
out, err := runScriptOnFile(state.a, item.Script, item.ScriptOptions, fi, fn)
156157
if err != nil {
157158
state.a.AddData(item.Name, fmt.Sprintf("DataExtract ERROR: script error: %s : %s : %s", err, item.Name, item.Desc))
158159
} else {
@@ -186,14 +187,19 @@ func (state *dataExtractType) CheckFile(fi *fsparser.FileInfo, filepath string)
186187
return nil
187188
}
188189

189-
// runScriptOnFile runs the provided script with the following parameters: <filename> <filename in filesystem> <uid> <gid> <mode> <selinux label - can be empty>
190-
func runScriptOnFile(a analyzer.AnalyzerType, script string, fi *fsparser.FileInfo, fpath string) (string, error) {
190+
// runScriptOnFile runs the provided script with the following parameters:
191+
// <filename> <filename in filesystem> <uid> <gid> <mode> <selinux label - can be empty> -- scriptOptions[0] scriptOptions[1]
192+
func runScriptOnFile(a analyzer.AnalyzerType, script string, scriptOptions []string, fi *fsparser.FileInfo, fpath string) (string, error) {
191193
fname, err := a.FileGet(fpath)
192194
if err != nil {
193195
return "", err
194196
}
195-
out, err := exec.Command(script, fname, filepath.Base(fpath),
196-
fmt.Sprintf("%d", fi.Uid), fmt.Sprintf("%d", fi.Gid), fmt.Sprintf("%o", fi.Mode), fi.SELinuxLabel).Output()
197+
options := []string{fname, filepath.Base(fpath), fmt.Sprintf("%d", fi.Uid), fmt.Sprintf("%d", fi.Gid), fmt.Sprintf("%o", fi.Mode), fi.SELinuxLabel}
198+
if len(scriptOptions) > 0 {
199+
options = append(options, "--")
200+
options = append(options, scriptOptions...)
201+
}
202+
out, err := exec.Command(script, options...).Output()
197203
_ = a.RemoveFile(fname)
198204

199205
return string(out), err

pkg/analyzer/filecmp/filecmp.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,7 @@ type cmpType struct {
3535
File string // filename
3636
OldFilePath string
3737
Script string
38-
ScriptOptions string
38+
ScriptOptions []string
3939
InformationalOnly bool // put result into Informational (not Offenders)
4040
name string // name of this check (need to be unique)
4141

@@ -163,7 +163,7 @@ func (state *fileCmpType) CheckFile(fi *fsparser.FileInfo, filepath string) erro
163163
args := []string{fi.Name, oldTmp, tmpfn}
164164
if len(item.ScriptOptions) > 0 {
165165
args = append(args, "--")
166-
args = append(args, item.ScriptOptions)
166+
args = append(args, item.ScriptOptions...)
167167
}
168168

169169
out, err := exec.Command(item.Script, args...).CombinedOutput()

pkg/analyzer/filecmp/filecmp_test.go

+3-3
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,7 @@ func TestCmp(t *testing.T) {
6565
[FileCmp."Test1"]
6666
File ="/cmp_test_1"
6767
Script = "diff.sh"
68-
ScriptOptions = ""
68+
ScriptOptions = [""]
6969
OldFilePath = "/tmp/analyzer_filecmp_1"
7070
`
7171

@@ -157,7 +157,7 @@ func TestCmpInfo(t *testing.T) {
157157
[FileCmp."Test1"]
158158
File ="/cmp_test_1"
159159
Script = "diff.sh"
160-
ScriptOptions = ""
160+
ScriptOptions = [""]
161161
InformationalOnly = true
162162
OldFilePath = "/tmp/analyzer_filecmp_1"
163163
`
@@ -220,7 +220,7 @@ func TestCmpNoOld(t *testing.T) {
220220
[FileCmp."Test1"]
221221
File ="/cmp_test_1"
222222
Script = "diff.sh"
223-
ScriptOptions = ""
223+
ScriptOptions = [""]
224224
OldFilePath = "/tmp/analyzer_filecmp_99"
225225
`
226226

pkg/analyzer/filecontent/filecontent.go

+6-6
Original file line numberDiff line numberDiff line change
@@ -262,14 +262,14 @@ type callbackDataType struct {
262262
* The script output is used to indicate an issue, the output is saved in the offender record.
263263
*
264264
* The first element in scriptOptions (from the callback data) defines a path match string.
265-
* This allows to specify a pattern the filename has to match. Files with names that do not patch will
266-
* be analyzed by the script. This is to speed up execution time since files have to be extracted
265+
* This allows to specify a pattern the filename has to match. Files with names that do not match will
266+
* not be analyzed by the script. This is to speed up execution time since files have to be extracted
267267
* to analyze them with the external script.
268268
*
269-
* The second element in scriptOptions will be passed to the script as cmd line arguments.
269+
* The following elements in scriptOptions will be passed to the script as cmd line arguments.
270270
*
271271
* The script is run with the following parameters:
272-
* script.sh <filename> <filename in filesystem> <uid> <gid> <mode> <selinux label - can be empty> -- <ScriptOptions[1]>
272+
* script.sh <filename> <filename in filesystem> <uid> <gid> <mode> <selinux label - can be empty> -- <ScriptOptions[1]> <ScriptOptions[2]>
273273
*/
274274
func checkFileScript(fi *fsparser.FileInfo, fullpath string, cbData analyzer.AllFilesCallbackData) {
275275
cbd := cbData.(*callbackDataType)
@@ -293,9 +293,9 @@ func checkFileScript(fi *fsparser.FileInfo, fullpath string, cbData analyzer.All
293293

294294
fname, _ := cbd.state.a.FileGet(path.Join(fullpath, fi.Name))
295295
args := []string{fname, fi.Name, fmt.Sprintf("%d", fi.Uid), fmt.Sprintf("%d", fi.Gid), fmt.Sprintf("%o", fi.Mode), fi.SELinuxLabel}
296-
if len(cbd.scriptOptions) >= 2 && len(cbd.scriptOptions[1]) > 0 {
296+
if len(cbd.scriptOptions) >= 2 {
297297
args = append(args, "--")
298-
args = append(args, cbd.scriptOptions[1])
298+
args = append(args, cbd.scriptOptions[1:]...)
299299
}
300300

301301
out, err := exec.Command(cbd.script, args...).CombinedOutput()

test/script_test.sh

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
#!/bin/bash
2+
3+
FILEPATH=$1
4+
ORIG_FILENAME=$2
5+
ORIG_UID=$3
6+
ORIG_GID=$4
7+
ORIG_MODE=$5
8+
ORIG_SELINUXLABEL=$6
9+
10+
# this is an artificial test
11+
if [ "$7" = "--" ]; then
12+
echo -n $9 $8
13+
fi

test/test.py

+19-13
Original file line numberDiff line numberDiff line change
@@ -45,28 +45,34 @@ def test(cfgfile, e2toolspath=""):
4545
if data.get("data", {}).get("Version") != "1.2.3":
4646
SetError("Data Version")
4747

48+
if data.get("data", {}).get("extract_test") != "test extract":
49+
SetError("extract test")
50+
4851
if "offenders" not in data:
4952
SetError("offenders")
5053
else:
51-
if "/dir2/file21" not in data["offenders"]:
54+
if not "/dir2/file21" in data["offenders"]:
5255
SetError("dir2/file21")
5356

54-
if "/dir2/file22" not in data["offenders"]:
57+
if not "/dir2/file22" in data["offenders"]:
5558
SetError("dir2/file22")
5659

57-
if data["offenders"]["/world"][0].find("WorldWriteable") == -1:
60+
if not "File is WorldWriteable, not allowed" in data["offenders"]["/world"]:
5861
SetError("WorldWriteable")
5962

60-
if data["offenders"]["/dir2/file21"][0].find("File is SUID") == -1:
63+
if not "File is SUID, not allowed" in data["offenders"]["/dir2/file21"]:
6164
SetError("SUID")
6265

63-
if data["offenders"]["/dir2/file22"][0].find("DirContent") == -1:
66+
if not "DirContent: File file22 not allowed in directory /dir2" in data["offenders"]["/dir2/file22"]:
6467
SetError("DirContent")
6568

66-
if "nofile" not in data["offenders"]:
69+
if not "nofile" in data["offenders"]:
6770
SetError("DirContent")
6871

69-
if data["offenders"]["/ver"][0].find("Digest") == -1:
72+
if not "test script" in data["offenders"]["/file2"]:
73+
SetError("file2")
74+
75+
if not "Digest (sha256) did not match found = 44c77e41961f354f515e4081b12619fdb15829660acaa5d7438c66fc3d326df3 should be = 8b15095ed1af38d5e383af1c4eadc5ae73cab03964142eb54cb0477ccd6a8dd4. ver needs to be specific : " in data["offenders"]["/ver"]:
7076
SetError("ver digest")
7177

7278
if "File State Check failed: group found 1002 should be 0 : this needs to be this way" in data["offenders"]["/dir2/file22"]:
@@ -78,21 +84,21 @@ def test(cfgfile, e2toolspath=""):
7884
if not "File State Check failed: size: 0 AllowEmpyt=false : this needs to be this way" in data["offenders"]["/file1"]:
7985
SetError("file1 exists but size 0")
8086

81-
if "elf_x8664 is not stripped" not in data["offenders"]["/bin/elf_x8664"]:
87+
if not "elf_x8664 is not stripped" in data["offenders"]["/bin/elf_x8664"]:
8288
SetError("script failed")
8389

84-
if "informational" not in data:
90+
if not "informational" in data:
8591
SetError("informational")
8692
else:
87-
if "/file1" not in data["informational"]:
93+
if not "/file1" in data["informational"]:
8894
SetError("/file1")
8995
else:
90-
if "changed" not in data["informational"]["/file1"][0]:
96+
if not "changed" in data["informational"]["/file1"][0]:
9197
SetError("file1 not changed")
92-
if "/date1" not in data["informational"]:
98+
if not "/date1" in data["informational"]:
9399
SetError("/date1")
94100
else:
95-
if "changed" not in data["informational"]["/date1"][0]:
101+
if not "changed" in data["informational"]["/date1"][0]:
96102
SetError("date1 not changed")
97103

98104
if __name__ == "__main__":

test/test_cfg.base.toml

+11
Original file line numberDiff line numberDiff line change
@@ -56,8 +56,19 @@ Script="check_file_x8664.sh"
5656

5757
[FileContent."ensure bins are stripped"]
5858
File = "/"
59+
ScriptOptions = ["*"]
5960
Script="check_file_elf_stripped.sh"
6061

62+
[FileContent."script_test"]
63+
File = "/file2"
64+
ScriptOptions = ["*", "script", "test"]
65+
Script="script_test.sh"
66+
67+
[DataExtract."extract_test"]
68+
File = "/file2"
69+
ScriptOptions = ["extract", "test"]
70+
Script="script_test.sh"
71+
6172
[FileContent."date1 needs to be specific"]
6273
File = "/date1"
6374
Digest="8b15095ed1af38d5e383af1c4eadc5ae73cab03964142eb54cb0477ccd6a8dd5"

0 commit comments

Comments
 (0)