Store logs in sqlite3 database
This commit is contained in:
		
							
								
								
									
										12
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								main.go
									
									
									
									
									
								
							| @@ -94,6 +94,10 @@ func main() { | |||||||
| 	kcfgPathFlag := app.Flag("kernels", "Path to main kernels config") | 	kcfgPathFlag := app.Flag("kernels", "Path to main kernels config") | ||||||
| 	kcfgPath := kcfgPathFlag.Default(defaultKcfgPath).String() | 	kcfgPath := kcfgPathFlag.Default(defaultKcfgPath).String() | ||||||
|  |  | ||||||
|  | 	defaultDbPath := usr.HomeDir + "/.out-of-tree/db.sqlite" | ||||||
|  | 	dbPathFlag := app.Flag("db", "Path to database") | ||||||
|  | 	dbPath := dbPathFlag.Default(defaultDbPath).String() | ||||||
|  |  | ||||||
| 	defaultUserKcfgPath := usr.HomeDir + "/.out-of-tree/kernels.user.toml" | 	defaultUserKcfgPath := usr.HomeDir + "/.out-of-tree/kernels.user.toml" | ||||||
| 	userKcfgPathFlag := app.Flag("user-kernels", "User kernels config") | 	userKcfgPathFlag := app.Flag("user-kernels", "User kernels config") | ||||||
| 	userKcfgPathEnv := userKcfgPathFlag.Envar("OUT_OF_TREE_KCFG") | 	userKcfgPathEnv := userKcfgPathFlag.Envar("OUT_OF_TREE_KCFG") | ||||||
| @@ -204,11 +208,17 @@ func main() { | |||||||
|  |  | ||||||
| 	handleFallbacks(kcfg) | 	handleFallbacks(kcfg) | ||||||
|  |  | ||||||
|  | 	db, err := openDatabase(*dbPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Fatalln(err) | ||||||
|  | 	} | ||||||
|  | 	defer db.Close() | ||||||
|  |  | ||||||
| 	switch kingpin.MustParse(app.Parse(os.Args[1:])) { | 	switch kingpin.MustParse(app.Parse(os.Args[1:])) { | ||||||
| 	case pewCommand.FullCommand(): | 	case pewCommand.FullCommand(): | ||||||
| 		err = pewHandler(kcfg, *path, *pewKernel, *pewBinary, | 		err = pewHandler(kcfg, *path, *pewKernel, *pewBinary, | ||||||
| 			*pewTest, *pewGuess, *qemuTimeout, *dockerTimeout, | 			*pewTest, *pewGuess, *qemuTimeout, *dockerTimeout, | ||||||
| 			*pewMax) | 			*pewMax, db) | ||||||
| 	case kernelListCommand.FullCommand(): | 	case kernelListCommand.FullCommand(): | ||||||
| 		err = kernelListHandler(kcfg) | 		err = kernelListHandler(kcfg) | ||||||
| 	case kernelAutogenCommand.FullCommand(): | 	case kernelAutogenCommand.FullCommand(): | ||||||
|   | |||||||
							
								
								
									
										71
									
								
								pew.go
									
									
									
									
									
								
							
							
						
						
									
										71
									
								
								pew.go
									
									
									
									
									
								
							| @@ -5,6 +5,7 @@ | |||||||
| package main | package main | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"database/sql" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"io/ioutil" | 	"io/ioutil" | ||||||
| @@ -16,6 +17,7 @@ import ( | |||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
|  | 	_ "github.com/mattn/go-sqlite3" | ||||||
| 	"github.com/otiai10/copy" | 	"github.com/otiai10/copy" | ||||||
| 	"github.com/remeh/sizedwaitgroup" | 	"github.com/remeh/sizedwaitgroup" | ||||||
| 	"gopkg.in/logrusorgru/aurora.v1" | 	"gopkg.in/logrusorgru/aurora.v1" | ||||||
| @@ -139,8 +141,15 @@ func genOkFail(name string, ok bool) (aurv aurora.Value) { | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
|  |  | ||||||
|  | type phasesResult struct { | ||||||
|  | 	Build, Run, Test struct { | ||||||
|  | 		Output string | ||||||
|  | 		Ok     bool | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| func dumpResult(q *qemu.QemuSystem, ka config.Artifact, ki config.KernelInfo, | func dumpResult(q *qemu.QemuSystem, ka config.Artifact, ki config.KernelInfo, | ||||||
| 	buildOk, runOk, testOk *bool) { | 	res *phasesResult, db *sql.DB) { | ||||||
|  |  | ||||||
| 	distroInfo := fmt.Sprintf("%s-%s {%s}", ki.DistroType, | 	distroInfo := fmt.Sprintf("%s-%s {%s}", ki.DistroType, | ||||||
| 		ki.DistroRelease, ki.KernelRelease) | 		ki.DistroRelease, ki.KernelRelease) | ||||||
| @@ -148,13 +157,13 @@ func dumpResult(q *qemu.QemuSystem, ka config.Artifact, ki config.KernelInfo, | |||||||
| 	colored := "" | 	colored := "" | ||||||
| 	if ka.Type == config.KernelExploit { | 	if ka.Type == config.KernelExploit { | ||||||
| 		colored = aurora.Sprintf("[*] %40s: %s %s", distroInfo, | 		colored = aurora.Sprintf("[*] %40s: %s %s", distroInfo, | ||||||
| 			genOkFail("BUILD", *buildOk), | 			genOkFail("BUILD", res.Build.Ok), | ||||||
| 			genOkFail("LPE", *testOk)) | 			genOkFail("LPE", res.Test.Ok)) | ||||||
| 	} else { | 	} else { | ||||||
| 		colored = aurora.Sprintf("[*] %40s: %s %s %s", distroInfo, | 		colored = aurora.Sprintf("[*] %40s: %s %s %s", distroInfo, | ||||||
| 			genOkFail("BUILD", *buildOk), | 			genOkFail("BUILD", res.Build.Ok), | ||||||
| 			genOkFail("INSMOD", *runOk), | 			genOkFail("INSMOD", res.Run.Ok), | ||||||
| 			genOkFail("TEST", *testOk)) | 			genOkFail("TEST", res.Test.Ok)) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	additional := "" | 	additional := "" | ||||||
| @@ -169,11 +178,16 @@ func dumpResult(q *qemu.QemuSystem, ka config.Artifact, ki config.KernelInfo, | |||||||
| 	} else { | 	} else { | ||||||
| 		fmt.Println(colored) | 		fmt.Println(colored) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	err := addToLog(db, q, ka, ki, res) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Println("[db] addToLog (", ka, ") error:", err) | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func whatever(swg *sizedwaitgroup.SizedWaitGroup, ka config.Artifact, | func whatever(swg *sizedwaitgroup.SizedWaitGroup, ka config.Artifact, | ||||||
| 	ki config.KernelInfo, binaryPath, testPath string, | 	ki config.KernelInfo, binaryPath, testPath string, | ||||||
| 	qemuTimeout, dockerTimeout time.Duration) { | 	qemuTimeout, dockerTimeout time.Duration, db *sql.DB) { | ||||||
|  |  | ||||||
| 	defer swg.Done() | 	defer swg.Done() | ||||||
|  |  | ||||||
| @@ -199,23 +213,21 @@ func whatever(swg *sizedwaitgroup.SizedWaitGroup, ka config.Artifact, | |||||||
| 	} | 	} | ||||||
| 	defer os.RemoveAll(tmp) | 	defer os.RemoveAll(tmp) | ||||||
|  |  | ||||||
| 	buildOk := false | 	result := phasesResult{} | ||||||
| 	runOk := false | 	defer dumpResult(q, ka, ki, &result, db) | ||||||
| 	testOk := false |  | ||||||
| 	defer dumpResult(q, ka, ki, &buildOk, &runOk, &testOk) |  | ||||||
|  |  | ||||||
| 	var outFile, output string | 	var outFile, output string | ||||||
| 	if binaryPath == "" { | 	if binaryPath == "" { | ||||||
| 		// TODO Write build log to file or database | 		// TODO Write build log to file or database | ||||||
| 		outFile, output, err = build(tmp, ka, ki, dockerTimeout) | 		outFile, result.Build.Output, err = build(tmp, ka, ki, dockerTimeout) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Println(output) | 			log.Println(output) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 		buildOk = true | 		result.Build.Ok = true | ||||||
| 	} else { | 	} else { | ||||||
| 		outFile = binaryPath | 		outFile = binaryPath | ||||||
| 		buildOk = true | 		result.Build.Ok = true | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	err = cleanDmesg(q) | 	err = cleanDmesg(q) | ||||||
| @@ -250,21 +262,19 @@ func whatever(swg *sizedwaitgroup.SizedWaitGroup, ka config.Artifact, | |||||||
|  |  | ||||||
| 	switch ka.Type { | 	switch ka.Type { | ||||||
| 	case config.KernelModule: | 	case config.KernelModule: | ||||||
| 		// TODO Write insmod log to file or database | 		result.Run.Output, err = q.CopyAndInsmod(outFile) | ||||||
| 		output, err := q.CopyAndInsmod(outFile) |  | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Println(output, err) | 			log.Println(result.Run.Output, err) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 		runOk = true | 		result.Run.Ok = true | ||||||
|  |  | ||||||
| 		// TODO Write test results to file or database | 		result.Test.Output, err = testKernelModule(q, ka, remoteTest) | ||||||
| 		output, err = testKernelModule(q, ka, remoteTest) |  | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Println(output, err) | 			log.Println(output, err) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 		testOk = true | 		result.Test.Ok = true | ||||||
| 	case config.KernelExploit: | 	case config.KernelExploit: | ||||||
| 		remoteExploit := fmt.Sprintf("/tmp/exploit_%d", rand.Int()) | 		remoteExploit := fmt.Sprintf("/tmp/exploit_%d", rand.Int()) | ||||||
| 		err = q.CopyFile("user", outFile, remoteExploit) | 		err = q.CopyFile("user", outFile, remoteExploit) | ||||||
| @@ -272,14 +282,14 @@ func whatever(swg *sizedwaitgroup.SizedWaitGroup, ka config.Artifact, | |||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// TODO Write test results to file or database | 		result.Test.Output, err = testKernelExploit(q, ka, remoteTest, | ||||||
| 		output, err = testKernelExploit(q, ka, remoteTest, remoteExploit) | 			remoteExploit) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Println(output) | 			log.Println(result.Test.Output) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 		runOk = true // does not really used | 		result.Run.Ok = true // does not really used | ||||||
| 		testOk = true | 		result.Test.Ok = true | ||||||
| 	default: | 	default: | ||||||
| 		log.Println("Unsupported artifact type") | 		log.Println("Unsupported artifact type") | ||||||
| 	} | 	} | ||||||
| @@ -296,7 +306,7 @@ func shuffleKernels(a []config.KernelInfo) []config.KernelInfo { | |||||||
|  |  | ||||||
| func performCI(ka config.Artifact, kcfg config.KernelConfig, binaryPath, | func performCI(ka config.Artifact, kcfg config.KernelConfig, binaryPath, | ||||||
| 	testPath string, qemuTimeout, dockerTimeout time.Duration, | 	testPath string, qemuTimeout, dockerTimeout time.Duration, | ||||||
| 	max int64) (err error) { | 	max int64, db *sql.DB) (err error) { | ||||||
|  |  | ||||||
| 	found := false | 	found := false | ||||||
|  |  | ||||||
| @@ -317,7 +327,7 @@ func performCI(ka config.Artifact, kcfg config.KernelConfig, binaryPath, | |||||||
| 			max -= 1 | 			max -= 1 | ||||||
| 			swg.Add() | 			swg.Add() | ||||||
| 			go whatever(&swg, ka, kernel, binaryPath, testPath, | 			go whatever(&swg, ka, kernel, binaryPath, testPath, | ||||||
| 				qemuTimeout, dockerTimeout) | 				qemuTimeout, dockerTimeout, db) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	swg.Wait() | 	swg.Wait() | ||||||
| @@ -371,7 +381,7 @@ func genAllKernels() (sk []config.KernelMask, err error) { | |||||||
| func pewHandler(kcfg config.KernelConfig, | func pewHandler(kcfg config.KernelConfig, | ||||||
| 	workPath, ovrrdKrnl, binary, test string, guess bool, | 	workPath, ovrrdKrnl, binary, test string, guess bool, | ||||||
| 	qemuTimeout, dockerTimeout time.Duration, | 	qemuTimeout, dockerTimeout time.Duration, | ||||||
| 	max int64) (err error) { | 	max int64, db *sql.DB) (err error) { | ||||||
|  |  | ||||||
| 	ka, err := config.ReadArtifactConfig(workPath + "/.out-of-tree.toml") | 	ka, err := config.ReadArtifactConfig(workPath + "/.out-of-tree.toml") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -399,7 +409,8 @@ func pewHandler(kcfg config.KernelConfig, | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	err = performCI(ka, kcfg, binary, test, qemuTimeout, dockerTimeout, max) | 	err = performCI(ka, kcfg, binary, test, qemuTimeout, dockerTimeout, | ||||||
|  | 		max, db) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user