av/revid/config/parameter/generate_parameters.go

313 lines
9.9 KiB
Go
Raw Permalink Normal View History

// +build ignore
/*
DESCRIPTION
generate_parameters.go uses a template to generate implementations for the
Parameter interface.
LICENSE
Copyright (C) 2020 the Australian Ocean Lab (AusOcean)
It is free software: you can redistribute it and/or modify them
under the terms of the GNU General Public License as published by the
Free Software Foundation, either version 3 of the License, or (at your
option) any later version.
It is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
for more details.
You should have received a copy of the GNU General Public License
along with revid in gpl.txt. If not, see http://www.gnu.org/licenses.
*/
package main
import (
"bytes"
"fmt"
"go/format"
"math"
"os"
"strings"
"text/template"
)
// Filename to house Parameter interface implementations.
const fileName = "parameters.go"
// Param holds characteristics for describing Parameter interface implementations.
type Param struct {
// N is the name of the struct implementing the Parameter interface.
N string
// R is the name of the receiver for the Type and Set methods. This is set to
// the lowercase of the first letter of the Parameter implementation name, N.
R string
// BT is the base type of a Parameter implementation. For example, for a
// Parameter implementation named Bitrate, i.e. type Bitrate uint, the base
// type is uint.
BT string
// E are the enums of a type i.e. if there is a list of identifiers we wish
// to recocgnise, they are specified here. For example, a Parameter implementation
// named Output may take on a value of HTTP, RTMP, RTP or File, so E is set to
// []string{"HTTP","RTMP","RTP","File"} and a const list of type Output will be
// generated as a result:
// const (
// OutputHTTP Output = iota
// OutputRTMP
// OutputRTP
// OutputFile
// )
E []string
// M, if defined, indicates a "multiple option type", which is to mean a slice
// of another type that is defined. For example, if the base type, BT, is set
// to a slice of a type "[]Output" (where Output is also defined in the params
// list), then M would be set to "Output". The E field must be manually set to
// be consistent with the Enums defined for the Output type.
M string
// If we wish an int, uint, or float64 value to be constrained to a particular
// range then Min and Max are both set to indicate the inclusive Min and Max
// possible values for a type. This will result in the generation of a range
// check in the implementation's Set method.
Min, Max int
}
// NB: Alphabetical order.
var params = []Param{
{N: "AutoWhiteBalance", BT: "uint8", E: []string{"Off", "Auto", "Sun", "Cloud", "Shade", "Tungsten", "Fluorescent", "Incandescent", "Flash", "Horizon"}},
{N: "BitDepth", BT: "uint"}, // TODO(Trek): bounds.
{N: "Bitrate", BT: "uint", Min: 1000, Max: 10000000},
{N: "Brightness", BT: "uint", Min: 0, Max: 100},
{N: "BurstPeriod", BT: "time.Duration"},
{N: "CBR", BT: "bool"},
{N: "CameraChan", BT: "uint8", E: []string{"Channel1", "Channel2"}},
{N: "CameraIP", BT: "string"},
{N: "Channels", BT: "uint"}, // TODO(Trek): bounds.
{N: "ClipDuration", BT: "time.Duration"},
{N: "Codec", BT: "uint8", E: []string{"H264", "H265", "MJPEG", "PCM", "ADPCM"}},
{N: "Exposure", BT: "uint8", E: []string{"Auto", "Night", "NightPreview", "BackLight", "SpotLight", "Sports", "Snow", "Beach", "VeryLong", "FixedFPS", "AntiShake", "Fireworks"}},
{N: "FileFPS", BT: "uint", Min: 1, Max: 30},
{N: "Filter", BT: "uint8", E: []string{"NoOp", "MOG", "VariableFPS", "KNN", "Difference", "Basic"}},
{N: "FrameRate", BT: "uint", Min: 1, Max: 30},
{N: "HTTPAddress", BT: "string"},
{N: "Height", BT: "uint", Min: 360, Max: 1080},
{N: "HorizontalFlip", BT: "bool"},
{N: "Input", BT: "uint8", E: []string{"File", "Raspivid", "Webcam", "RTSP"}},
{N: "InputPath", BT: "string"},
{N: "Level", BT: "uint8", E: []string{"Debug", "Info", "Warning", "Error", "Fatal"}},
{N: "MinFPS", BT: "uint", Min: 1, Max: 30},
{N: "MinFrames", BT: "uint", Min: 0, Max: 1000},
{N: "Mode", BT: "uint8", E: []string{"Normal", "Paused", "Burst", "Loop"}},
{N: "MotionDownscaling", BT: "uint"}, // TODO(Scott): define bounds.
{N: "MotionHistory", BT: "uint"}, // TODO(Scott/Ella): define bounds.
{N: "MotionInterval", BT: "uint", Min: 0, Max: 30},
{N: "MotionKernel", BT: "uint"}, // TODO(Scott/Ella): define bounds.
{N: "MotionMinArea", BT: "float64"}, // TODO(Scott/Ella): define bounds.
{N: "MotionPixels", BT: "uint"}, // TODO(Scott/Ella): define bounds.
{N: "MotionThreshold", BT: "float64"}, // TODO(Scott/Ella): define bounds.
{N: "Output", BT: "uint8", E: []string{"HTTP", "RTMP", "RTP", "File"}},
{N: "OutputPath", BT: "string"},
{N: "Outputs", BT: "[]Output", M: "Output", E: []string{"HTTP", "RTMP", "RTP", "File"}},
{N: "PSITime", BT: "time.Duration"},
{N: "Quantization", BT: "uint"},
{N: "RBCapacity", BT: "uint", Min: 1000000, Max: 100000000},
{N: "RBMaxElements", BT: "uint", Min: 0, Max: math.MaxUint32},
{N: "RBWriteTimeout", BT: "time.Duration"},
{N: "RTMPURL", BT: "string"},
{N: "RTPAddress", BT: "string"},
{N: "RecPeriod", BT: "float64"}, // TODO(Trek): bounds.
{N: "Rotation", BT: "uint", Min: 0, Max: 359},
{N: "SampleRate", BT: "uint"}, // TODO(Trek): bounds.
{N: "Saturation", BT: "int", Min: -50, Max: 50},
{N: "ShowWindows", BT: "bool"},
{N: "VBRBitrate", BT: "uint", Min: 1, Max: 30},
{N: "VBRQuality", BT: "uint8", E: []string{"Standard", "Fair", "Good", "Great", "Excellent"}},
{N: "VerticalFlip", BT: "bool"},
{N: "Width", BT: "uint", Min: 640, Max: 1920},
{N: "WriteRate", BT: "float64"},
}
const fileHeader = `
/*
DESCRIPTION
Code generated by "go run generate_parameters.go; DO NOT EDIT.
parameters.go contains implementations of the Parameter interface for all
parameter types required by the configuration struct.
AUTHORS
Saxon Nelson-Milton <saxon@ausocean.org>
LICENSE
Copyright (C) 2020 the Australian Ocean Lab (AusOcean)
It is free software: you can redistribute it and/or modify them
under the terms of the GNU General Public License as published by the
Free Software Foundation, either version 3 of the License, or (at your
option) any later version.
It is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
for more details.
You should have received a copy of the GNU General Public License
along with revid in gpl.txt. If not, see http://www.gnu.org/licenses.
*/
package parameter
import (
"fmt"
"strconv"
"time"
"strings"
)
type Parameter interface {
Type() string
Set(val string) error
}
`
const paramTemplate = `
type {{.N}} {{.BT}}
{{if and .E (not .M)}}
const (
{{$name := .N}}
{{- range $i, $e := .E}}{{- if eq $i 0}}{{$name}}{{$e}} {{$name}} = iota{{else}}{{$name}}{{$e}}{{end}}
{{end -}}
)
{{end -}}
{{- if .E}}
func ({{.R}} *{{.N}}) Type() string { return "enum:{{range $i, $e := .E}}{{if eq $i 0}}{{$e}}{{else}},{{$e}}{{end}}{{end}}"}
{{else}}
func ({{.R}} *{{.N}}) Type() string { return "{{.BT}}"}
{{end -}}
func ({{.R}} *{{.N}}) Set(val string) error {
{{- if eq .BT "string"}}
*{{.R}} = {{.N}}(val)
{{else if eq .BT "bool"}}
switch val {
case "true":
*{{.R}} = true
case "false":
*{{.R}} = false
default:
return fmt.Errorf("not boolean value: %s",val)
}
{{else if eq .BT "int"}}
_v, err := strconv.Atoi(val)
if err != nil {
return fmt.Errorf("could not convert set string to int: %w",err)
}
{{if ne .Min .Max}}
if _v < {{.Min}} || _v > {{.Max}} {
return fmt.Errorf("invalid value %v",_v)
}
{{end}}
*{{.R}} = {{.N}}(_v)
{{else if eq .BT "uint"}}
_v, err := strconv.Atoi(val)
if err != nil {
return fmt.Errorf("could not convert set string to int: %w",err)
}
{{if ne .Min .Max}}
if _v < {{.Min}} || _v > {{.Max}} {
return fmt.Errorf("invalid value %v",_v)
}
{{end}}
*{{.R}} = {{.N}}(_v)
{{else if eq .BT "float64"}}
_v, err := strconv.ParseFloat(val,64)
if err != nil {
return fmt.Errorf("could not convert set string to float: %w",err)
}
{{if ne .Min .Max}}
if _v < {{.Min}} || _v > {{.Max}} {
return fmt.Errorf("invalid value %v",_v)
}
{{end}}
*{{.R}} = {{.N}}(_v)
{{else if eq .BT "time.Duration"}}
_v, err := strconv.Atoi(val)
if err != nil {
return fmt.Errorf("could not convert set string to int: %w",err)
}
*{{.R}} = {{.N}}(time.Duration(_v)*time.Second)
{{else if .M}}
vals := strings.Split(val, ",")
*{{.R}} = make({{.BT}}, len(vals))
for i, v := range vals {
switch v {
{{- $receiver := .R}}
{{- $m := .M}}
{{range .E}}case "{{ . }}":
(*{{$receiver}})[i] = {{$m}}{{ . }}
{{end -}}
default:
return fmt.Errorf("unrecognised {{.N}}: %s",val)
}
}
{{else}}
switch val {
{{- $receiver := .R}}
{{- $name := .N}}
{{range .E}}case "{{ . }}":
*{{$receiver}} = {{$name}}{{ . }}
{{end -}}
default:
return fmt.Errorf("unrecognised {{.N}}: %s",val)
}
{{end -}}
return nil
}
`
func main() {
f, err := os.Create(fileName)
if err != nil {
panic(fmt.Sprintf("error creating %s file: %v", fileName, err))
}
defer f.Close()
var buf bytes.Buffer
_, err = buf.Write([]byte(fileHeader))
if err != nil {
panic(fmt.Sprintf("error writing header: %v", err))
}
param := template.Must(template.New("param").Parse(paramTemplate))
for _, p := range params {
// Use first letter of parameter name as receiver.
p.R = strings.ToLower(p.N[:1])
err = param.Execute(&buf, p)
if err != nil {
panic(fmt.Sprintf("error executing template: %v", err))
}
}
b, err := format.Source(buf.Bytes())
if err != nil {
f.Write(buf.Bytes()) // Useful to debug bad format.
panic(fmt.Sprintf("error formatting: %v", err))
}
_, err = f.Write(b)
if err != nil {
panic(fmt.Sprintf("error writing %s file: %v", fileName, err))
}
}