2020-02-12 03:27:08 +03:00
|
|
|
// +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)
|
|
|
|
}
|
2020-02-17 01:25:43 +03:00
|
|
|
{{if ne .Min .Max}}
|
|
|
|
if _v < {{.Min}} || _v > {{.Max}} {
|
|
|
|
return fmt.Errorf("invalid value %v",_v)
|
|
|
|
}
|
|
|
|
{{end}}
|
2020-02-12 03:27:08 +03:00
|
|
|
*{{.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)
|
|
|
|
}
|
2020-02-17 01:25:43 +03:00
|
|
|
{{if ne .Min .Max}}
|
|
|
|
if _v < {{.Min}} || _v > {{.Max}} {
|
|
|
|
return fmt.Errorf("invalid value %v",_v)
|
|
|
|
}
|
|
|
|
{{end}}
|
2020-02-12 03:27:08 +03:00
|
|
|
*{{.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)
|
|
|
|
}
|
2020-02-17 01:25:43 +03:00
|
|
|
{{if ne .Min .Max}}
|
|
|
|
if _v < {{.Min}} || _v > {{.Max}} {
|
|
|
|
return fmt.Errorf("invalid value %v",_v)
|
|
|
|
}
|
|
|
|
{{end}}
|
|
|
|
*{{.R}} = {{.N}}(_v)
|
2020-02-12 03:27:08 +03:00
|
|
|
{{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)
|
2020-02-17 01:25:43 +03:00
|
|
|
{{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)
|
|
|
|
}
|
|
|
|
}
|
2020-02-12 03:27:08 +03:00
|
|
|
{{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))
|
|
|
|
}
|
|
|
|
}
|