feat(train): add new command to interact with aws and train models

This commit is contained in:
2021-10-17 19:15:44 +02:00
parent 5436dfebc2
commit 538cea18f2
1064 changed files with 282251 additions and 89305 deletions

191
pkg/data/data.go Normal file
View File

@ -0,0 +1,191 @@
package data
import (
"archive/zip"
"bytes"
"encoding/json"
"fmt"
"github.com/cyrilix/robocar-tools/record"
log "github.com/sirupsen/logrus"
"io/ioutil"
"os"
"path"
"regexp"
)
var camSubDir = "cam"
func WriteArchive(basedir string, archiveName string, sliceSize int) error {
content, err := BuildArchive(basedir, sliceSize)
if err != nil {
return fmt.Errorf("unable to build archive: %w", err)
}
if err != nil {
return fmt.Errorf("unable to build archive: %w", err)
}
err = ioutil.WriteFile(archiveName, content, os.FileMode(0755))
if err != nil {
return fmt.Errorf("unable to write archive content to disk: %w", err)
}
return nil
}
func BuildArchive(basedir string, sliceSize int) ([]byte, error) {
log.Printf("build zip archive from %s\n", basedir)
dirItems, err := ioutil.ReadDir(basedir)
if err != nil {
return nil, fmt.Errorf("unable to list directory in %v dir: %w", basedir, err)
}
imgCams := make([]string, 0)
records := make([]string, 0)
for _, dirItem := range dirItems {
log.Debugf("process %v directory", dirItem)
imgDir := path.Join(basedir, dirItem.Name(), camSubDir)
imgs, err := ioutil.ReadDir(imgDir)
if err != nil {
return nil, fmt.Errorf("unable to list cam images in directory %v: %w", imgDir, err)
}
for _, img := range imgs {
idx, err := indexFromFile(img.Name())
if err != nil {
return nil, fmt.Errorf("unable to find index in cam image name %v: %w", img.Name(), err)
}
log.Debugf("found image with index %v", idx)
records = append(records, path.Join(basedir, dirItem.Name(), fmt.Sprintf(record.RecorNameFormat, idx)))
imgCams = append(imgCams, path.Join(basedir, dirItem.Name(), camSubDir, img.Name()))
}
}
if sliceSize > 0 {
imgCams, records, err = applySlice(imgCams, records, sliceSize)
}
content, err := buildArchiveContent(imgCams, records)
if err != nil {
return nil , fmt.Errorf("unable to generate archive content: %w", err)
}
log.Printf("archive built\n")
return content, nil
}
func applySlice(imgCams []string, records []string, sliceSize int) ([]string, []string, error) {
// Add sliceSize images shift
i := imgCams[:len(imgCams)-sliceSize]
r := records[sliceSize:]
return i, r, nil
}
var indexRegexp *regexp.Regexp
func init() {
re, err := regexp.Compile("image_array_(?P<idx>[0-9]+)\\.jpg$")
if err != nil {
log.Fatalf("unable to compile regex: %v", err)
}
indexRegexp = re
}
func indexFromFile(fileName string) (string, error) {
matches := findNamedMatches(indexRegexp, fileName)
if matches["idx"] == "" {
return "", fmt.Errorf("no index in filename")
}
return matches["idx"], nil
}
func findNamedMatches(regex *regexp.Regexp, str string) map[string]string {
match := regex.FindStringSubmatch(str)
results := map[string]string{}
for i, name := range match {
results[regex.SubexpNames()[i]] = name
}
return results
}
func buildArchiveContent(imgFiles []string, recordFiles []string) ([]byte, error) {
// Create a buffer to write our archive to.
buf := new(bytes.Buffer)
// Create a new zip archive.
w := zip.NewWriter(buf)
err := addJsonFiles(recordFiles, imgFiles, w)
if err != nil {
return nil, fmt.Errorf("unable to write json files in zip archive: %w", err)
}
err = addCamImages(imgFiles, w)
if err != nil {
return nil, fmt.Errorf("unable to cam files in zip archive: %w", err)
}
err = w.Close()
if err != nil {
return nil, fmt.Errorf("unable to build archive: %w", err)
}
content, err := ioutil.ReadAll(buf)
return content, err
}
func addCamImages(imgFiles []string, w *zip.Writer) error {
for _, img := range imgFiles {
imgContent, err := ioutil.ReadFile(img)
if err != nil {
return fmt.Errorf("unable to read img: %w", err)
}
_, imgName := path.Split(img)
err = addToArchive(w, imgName, imgContent)
if err != nil {
return fmt.Errorf("unable to create new img entry in archive: %w", err)
}
}
return nil
}
func addJsonFiles(recordFiles []string, imgCam []string, w *zip.Writer) error {
for idx, r := range recordFiles {
content, err := ioutil.ReadFile(r)
if err != nil {
return fmt.Errorf("unable to read json content: %w", err)
}
var rcd record.Record
err = json.Unmarshal(content, &rcd)
if err != nil {
return fmt.Errorf("unable to unmarshal record: %w", err)
}
_, camName := path.Split((imgCam)[idx])
rcd.CamImageArray = camName
recordBytes, err := json.Marshal(&rcd)
if err != nil {
return fmt.Errorf("unable to marshal %v record: %w", rcd, err)
}
_, recordName := path.Split(r)
err = addToArchive(w, recordName, recordBytes)
if err != nil {
return fmt.Errorf("unable to create new record in archive: %w", err)
}
}
return nil
}
func addToArchive(w *zip.Writer, name string, content []byte) error {
recordWriter, err := w.Create(name)
if err != nil {
return fmt.Errorf("unable to create new entry %v in archive: %w", name, err)
}
_, err = recordWriter.Write(content)
if err != nil {
return fmt.Errorf("unable to add content in %v zip archive: %w", name, err)
}
return nil
}

118
pkg/data/data_test.go Normal file
View File

@ -0,0 +1,118 @@
package data
import (
"archive/zip"
"encoding/json"
"fmt"
"github.com/cyrilix/robocar-tools/record"
log "github.com/sirupsen/logrus"
"io/ioutil"
"os"
"path"
"strings"
"testing"
)
func TestBuildArchive(t *testing.T) {
tmpDir, err := ioutil.TempDir("", "buildarchive")
if err != nil {
t.Fatalf("unable to make tmpdir: %v", err)
}
defer func() {
err := os.RemoveAll(tmpDir)
if err != nil {
log.Warnf("unable to remove tempdir %v: %v", tmpDir, err)
}
}()
archive := path.Join(tmpDir, "train.zip")
expectedRecordFiles, expectedImgFiles := expectedFiles()
err = WriteArchive("testdata", archive, 0)
if err != nil {
t.Errorf("unable to build archive: %v", err)
}
r, err := zip.OpenReader(archive)
if err != nil {
t.Errorf("unable to read archive, %v", err)
}
defer r.Close()
if len(r.File) != len(expectedImgFiles)+len(expectedRecordFiles) {
t.Errorf("bad number of files in archive: %v, wants %v", len(r.File), len(expectedImgFiles)+len(expectedRecordFiles))
}
// Iterate through the files in the archive,
// printing some of their contents.
for _, f := range r.File {
filename := f.Name
if filename[len(filename)-4:] == "json" {
expectedRecordFiles[filename] = true
expectedtImgName := strings.Replace(filename, "record", "cam-image_array", 1)
expectedtImgName = strings.Replace(expectedtImgName, "json", "jpg", 1)
checkJsonContent(t, f, expectedtImgName)
continue
}
if filename[len(filename)-3:] == "jpg" {
expectedImgFiles[filename] = true
continue
}
t.Errorf("unexpected file in archive: %v", filename)
}
checkAllFilesAreFoundInArchive(expectedRecordFiles, t, expectedImgFiles)
}
func checkAllFilesAreFoundInArchive(expectedRecordFiles map[string]bool, t *testing.T, expectedImgFiles map[string]bool) {
for f, found := range expectedRecordFiles {
if !found {
t.Errorf("%v not found in archive", f)
}
}
for f, found := range expectedImgFiles {
if !found {
t.Errorf("%v not found in archive", f)
}
}
}
func checkJsonContent(t *testing.T, f *zip.File, expectedCamImage string) {
rc, err := f.Open()
if err != nil {
t.Errorf("unable to read file content of %v: %v", f.Name, err)
}
defer rc.Close()
content, err := ioutil.ReadAll(rc)
if err != nil {
t.Errorf("%v has invalid json content: %v", f.Name, err)
}
var rcd record.Record
err = json.Unmarshal(content, &rcd)
if err != nil {
t.Errorf("unable to unmarshal json content of%v: %v", f.Name, err)
}
if rcd.CamImageArray != expectedCamImage {
t.Errorf("record %v: invalid image ref: %v, wants %v", f.Name, rcd.CamImageArray, expectedCamImage)
}
if rcd.UserAngle == 0. {
t.Errorf("record %v: user angle has not been initialised", f.Name)
}
}
func expectedFiles() (map[string]bool, map[string]bool) {
expectedRecordFiles := make(map[string]bool)
expectedImgFiles := make(map[string]bool)
for i := 1; i <= 8; i++ {
expectedRecordFiles[fmt.Sprintf("record_%07d.json", i)] = false
expectedImgFiles[fmt.Sprintf("cam-image_array_%07d.jpg", i)] = false
}
for i := 101; i <= 106; i++ {
expectedRecordFiles[fmt.Sprintf("record_%07d.json", i)] = false
expectedImgFiles[fmt.Sprintf("cam-image_array_%07d.jpg", i)] = false
}
return expectedRecordFiles, expectedImgFiles
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

View File

@ -0,0 +1 @@
{"user/angle":0.04117644,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000001.jpg"}

View File

@ -0,0 +1 @@
{"user/angle":0.045098066,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000002.jpg"}

View File

@ -0,0 +1 @@
{"user/angle":0.045098066,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000003.jpg"}

View File

@ -0,0 +1 @@
{"user/angle":0.04117644,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000004.jpg"}

View File

@ -0,0 +1 @@
{"user/angle":0.04117644,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000005.jpg"}

View File

@ -0,0 +1 @@
{"user/angle":0.043137312,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000006.jpg"}

View File

@ -0,0 +1 @@
{"user/angle":0.04117644,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000007.jpg"}

View File

@ -0,0 +1 @@
{"user/angle":0.045098066,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000008.jpg"}

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.5 KiB

View File

@ -0,0 +1 @@
{"user/angle":0.04117644,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000101.jpg"}

View File

@ -0,0 +1 @@
{"user/angle":0.045098066,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000102.jpg"}

View File

@ -0,0 +1 @@
{"user/angle":0.045098066,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000103.jpg"}

View File

@ -0,0 +1 @@
{"user/angle":0.04117644,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000104.jpg"}

View File

@ -0,0 +1 @@
{"user/angle":0.04117644,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000105.jpg"}

View File

@ -0,0 +1 @@
{"user/angle":0.043137312,"cam/image_array":"/tmp/record//2020021819-3/cam/cam-image_array_0000106.jpg"}