Compare commits

...

6 Commits
v0.2 ... main

6 changed files with 239 additions and 93 deletions

BIN
.DS_Store vendored Normal file

Binary file not shown.

16
Dockerfile Normal file
View File

@ -0,0 +1,16 @@
FROM golang:1.20
# Build app
WORKDIR /app
COPY go.mod go.sum ./
RUN go mod download
COPY *.go ./
RUN go build -o /midi-request-trigger
WORKDIR /app
RUN rm -Rf /app; mkdir /etc/midi-request-trigger
# Configuration volume
VOLUME ["/etc/midi-request-trigger"]
# Command
CMD ["/midi-request-trigger"]

View File

@ -110,7 +110,7 @@ On MacOS, there is an IAC Driver that can be enabled in Audio MIDI Setup.
midi_routers:
- name: service_notifications
device: IAC Driver Bus 1
debug_listener: true
log_level: 2
```
### Example note trigger configuration
@ -120,7 +120,7 @@ midi_routers:
midi_routers:
- name: service_notifications
device: IAC Driver Bus 1
debug_listener: true
log_level: 2
note_triggers:
- channel: 0
note: 0
@ -136,7 +136,7 @@ midi_routers:
midi_routers:
- name: service_notifications
device: IAC Driver Bus 1
debug_listener: true
log_level: 2
request_triggers:
- channel: 0
note: 0
@ -152,7 +152,7 @@ midi_routers:
midi_routers:
- name: service_notifications
device: IAC Driver Bus 1
debug_listener: true
log_level: 3
note_triggers:
- channel: 0
note: 0
@ -174,5 +174,85 @@ midi_routers:
headers:
Content-Type:
- multipart/form-data; boundary=---------------------------888832887744
debug_request: true
```
### Example mqtt config
```yaml
---
midi_routers:
- name: Wing Midi Signals
device: WING Port 4
mqtt:
host: 10.0.0.2
port: 1883
client_id: midi_mqtt_bridge
user: mqtt
password: password
topic: midi/behringer_wing
log_level: 4
note_triggers:
- channel: 0
note: 1
match_all_velocities: true
mqtt_topic: osc/behringer_wing/send/$ctl/user/2/2/enc/val
mqtt_payload:
- "1"
- channel: 0
note: 2
match_all_velocities: true
mqtt_topic: osc/behringer_wing/send/$ctl/user/2/2/enc/val
mqtt_payload:
- "2"
- channel: 0
note: 3
match_all_velocities: true
mqtt_topic: osc/behringer_wing/send/$ctl/user/2/2/enc/val
mqtt_payload:
- "3"
- channel: 0
note: 4
match_all_velocities: true
mqtt_topic: osc/behringer_wing/send/$ctl/user/2/2/enc/val
mqtt_payload:
- "4"
- channel: 0
note: 5
match_all_velocities: true
mqtt_topic: osc/behringer_wing/send/$ctl/user/2/2/enc/val
mqtt_payload:
- "5"
- channel: 0
note: 6
match_all_velocities: true
mqtt_topic: osc/behringer_wing/send/$ctl/user/2/2/enc/val
mqtt_payload:
- "6"
- channel: 0
note: 7
match_all_velocities: true
mqtt_topic: osc/behringer_wing/send/$ctl/user/2/2/enc/val
mqtt_payload:
- "7"
- channel: 0
note: 8
match_all_velocities: true
mqtt_topic: osc/behringer_wing/send/$ctl/user/2/2/enc/val
mqtt_payload:
- "8"
- channel: 0
match_all_notes: true
match_all_velocities: true
delay_before: 200ms
mqtt_topic: osc/behringer_wing/send/$ctl/user/2/2/bu/val
mqtt_payload:
- "1"
- channel: 0
match_all_notes: true
match_all_velocities: true
delay_before: 200ms
mqtt_topic: osc/behringer_wing/send/$ctl/user/2/2/bu/val
mqtt_payload:
- "0"
delay_after: 200ms
```

View File

@ -48,7 +48,7 @@ func (a *App) ReadConfig() {
} else if _, err := os.Stat(etcConfig); err == nil {
configFile = etcConfig
} else {
log.Fatal("Unable to find a configuration file.")
log.Println("Unable to find a configuration file.")
}
// Load the configuration file.
@ -68,6 +68,7 @@ func (a *App) ReadConfig() {
fig.Dirs(filePath),
)
if err != nil {
app.config = config
log.Printf("Error parsing configuration: %s\n", err)
return
}

View File

@ -15,7 +15,7 @@ import (
const (
serviceName = "midi-request-trigger"
serviceDescription = "Takes trigger MIDI messages by HTTP or MQTT requests and trigger HTTP or MQTT requests by MIDI messages"
serviceVersion = "0.2"
serviceVersion = "0.3"
)
// App is the global application structure for communicating between servers and storing information.

View File

@ -11,6 +11,7 @@ import (
"regexp"
"strconv"
"strings"
"time"
mqtt "github.com/eclipse/paho.mqtt.golang"
"gitlab.com/gomidi/midi/v2"
@ -21,8 +22,10 @@ import (
type LogLevel int
const (
// Logs only errors.
ErrorLog LogLevel = iota
// Logs info messages.
InfoLog LogLevel = iota
// Log only errors.
ErrorLog
// MQTT, HTTP, and MIDI receive logging.
ReceiveLog
// MQTT, HTTP, and MIDI send logging.
@ -33,7 +36,7 @@ const (
// Provides a string value for a log level.
func (l LogLevel) String() string {
return [...]string{"Error", "Receive", "Send", "Debug"}[l]
return [...]string{"Info", "Error", "Receive", "Send", "Debug"}[l]
}
// Configurations relating to MQTT connection.
@ -70,20 +73,25 @@ type MQTTPayload struct {
// Triggers that occur from MIDI messages received.
type NoteTrigger struct {
// If set, every note played will be matched.
MatchAllNotes bool `fig:"match_all_notes"`
// Channel to match.
Channel uint8 `fig:"channel"`
// If we should match all channel values.
MatchAllChannels bool `fig:"match_all_channels"`
// Note to match.
Note uint8 `fig:"note"`
// If we should match all note values.
MatchAllNotes bool `fig:"match_all_notes"`
// Velocity to match.
Velocity uint8 `fig:"velocity"`
// If we should match all velocity values.
MatchAllVelocities bool `fig:"match_all_velocities"`
// Allow delaying the request.
DelayBefore time.Duration `fig:"delay_before"`
DelayAfter time.Duration `fig:"deplay_after"`
// Custom MQTT message. Do not set to ignore MQTT.
MqttTopic string `fig:"mqtt_topic"`
// Nil payload will generate a payload with midi info.
MqttPayload []interface{} `fig:"mqtt_payload"`
MqttPayload interface{} `fig:"mqtt_payload"`
// If the HTTP request should includ midi info.
MidiInfoInRequest bool `fig:"midi_info_in_request"`
// Should SSL requests require a valid certificate.
@ -132,18 +140,19 @@ type MidiRouter struct {
RequestTriggers []RequestTrigger `fig:"request_triggers"`
// How much logging.
// 0 - Errors
// 1 - MQTT and OSC receive logging.
// 2 - MQTT and OSC send logging.
// 3 - Debug
LogLevel LogLevel `yaml:"log_level" json:"log_level"`
// 0 - Info
// 1 - Errors
// 2 - MQTT, HTTP, and MIDI receive logging.
// 3 - MQTT, HTTP, and MIDI send logging.
// 4 - Debug
LogLevel LogLevel `fig:"log_level"`
// Connection to MIDI device.
MidiOut drivers.Out `fig:"-"`
// Function to stop listening to MIDI device.
ListenerStop func() `fig:"-"`
// The client connection to MQTT.
MqttClient mqtt.Client `yaml:"-" json:"-"`
MqttClient mqtt.Client `fig:"-"`
}
// Logging function to allow log levels.
@ -156,7 +165,7 @@ func (r *MidiRouter) Log(level LogLevel, format string, args ...interface{}) {
// When a MIDI message occurs, send the HTTP request.
func (r *MidiRouter) sendRequest(channel, note, velocity uint8) {
// If MQTT firehose not disabled, send to general cmd topic.
if r.MQTT.Host != "" && r.MQTT.Port != 0 && !r.MQTT.DisableMidiFirehose {
if r.MqttClient != nil && !r.MQTT.DisableMidiFirehose {
payload := MQTTPayload{
Channel: channel,
Note: note,
@ -168,7 +177,7 @@ func (r *MidiRouter) sendRequest(channel, note, velocity uint8) {
} else {
topic := r.MQTT.Topic + "/cmd"
r.MqttClient.Publish(topic, 0, true, data)
r.Log(SendLog, "-> [MQTT] %s", topic)
r.Log(SendLog, "-> [MQTT] %s: %s", topic, string(data))
}
}
@ -177,15 +186,26 @@ func (r *MidiRouter) sendRequest(channel, note, velocity uint8) {
// If match all notes, process this request.
// If not, check if channel, note, and velocity matches.
// The velocity may be defined to accept all.
if trig.MatchAllNotes || (trig.Channel == channel && trig.Note == note && (trig.Velocity == velocity || trig.MatchAllVelocities)) {
if (trig.Channel == channel || trig.MatchAllChannels) && (trig.Note == note || trig.MatchAllNotes) && (trig.Velocity == velocity || trig.MatchAllVelocities) {
// For all logging, we want to print the message so setup a common string to print.
logInfo := fmt.Sprintf("note %s(%d) on channel %v with velocity %v", midi.Note(note), note, channel, velocity)
if trig.MqttTopic != "" {
// Delay before.
time.Sleep(trig.DelayBefore)
// If MQTT trigger, send the MQTT request.
if trig.MqttTopic != "" && r.MqttClient != nil {
// If payload provided, send the defined payload.
if trig.MqttPayload != nil {
r.MqttClient.Publish(trig.MqttTopic, 0, true, trig.MqttPayload)
r.Log(SendLog, "-> [MQTT] %s", trig.MqttTopic)
data, err := json.Marshal(trig.MqttPayload)
if err != nil {
r.Log(ErrorLog, "Json Encode: %s", err)
} else {
r.MqttClient.Publish(trig.MqttTopic, 0, true, data)
r.Log(SendLog, "-> [MQTT] %s: %s", trig.MqttTopic, string(data))
}
} else {
// If no payload provided, send the note information as JSON.
payload := MQTTPayload{
Channel: channel,
Note: note,
@ -196,11 +216,12 @@ func (r *MidiRouter) sendRequest(channel, note, velocity uint8) {
r.Log(ErrorLog, "Json Encode: %s", err)
} else {
r.MqttClient.Publish(trig.MqttTopic, 0, true, data)
r.Log(SendLog, "-> [MQTT] %s", trig.MqttTopic)
r.Log(SendLog, "-> [MQTT] %s: %s", trig.MqttTopic, string(data))
}
}
}
// If URL trigger defined, perform a HTTP request.
if trig.URL != "" {
// Default method to GET if nothing is defined.
if trig.Method == "" {
@ -269,6 +290,9 @@ func (r *MidiRouter) sendRequest(channel, note, velocity uint8) {
r.Log(DebugLog, "Trigger response: %s\n%s", logInfo, string(body))
}
}
// Delay after.
time.Sleep(trig.DelayAfter)
}
}
}
@ -455,81 +479,105 @@ func (r *MidiRouter) MqttSubscribe(topic string) {
func (r *MidiRouter) Connect() {
// If request triggers defined, find the out port.
if len(r.RequestTriggers) != 0 {
out, err := midi.FindOutPort(r.Device)
if err != nil {
log.Println("Can't find output device:", r.Device)
} else {
r.MidiOut = out
}
go func() {
for {
out, err := midi.FindOutPort(r.Device)
if err != nil {
r.Log(ErrorLog, "Can't find output device: %s", r.Device)
} else {
r.MidiOut = out
break
}
r.Log(ErrorLog, "Retrying in 1 minute.")
time.Sleep(time.Minute)
}
}()
}
// If listener is disabled, stop here.
if r.DisableListener {
return
if !r.DisableListener {
go func() {
for {
// Try finding input port.
r.Log(InfoLog, "Connecting to input device: %s", r.Device)
in, err := midi.FindInPort(r.Device)
if err != nil {
r.Log(ErrorLog, "Can't find input device: %s", r.Device)
r.Log(ErrorLog, "Retrying in 1 minute.")
time.Sleep(time.Minute)
continue
}
// Start listening to MIDI messages.
stop, err := midi.ListenTo(in, func(msg midi.Message, timestampms int32) {
var channel, note, velocity uint8
switch {
// Get notes with an velocity set.
case msg.GetNoteStart(&channel, &note, &velocity):
r.Log(ReceiveLog, "starting note %s(%d) on channel %v with velocity %v", midi.Note(note), note, channel, velocity)
// Process request.
r.sendRequest(channel, note, velocity)
// If no velocity is set, an note end message is received.
case msg.GetNoteEnd(&channel, &note):
r.Log(ReceiveLog, "ending note %s(%d) on channel %v", midi.Note(note), note, channel)
// Process request.
r.sendRequest(channel, note, 0)
default:
// ignore
}
})
if err != nil {
r.Log(ErrorLog, "Error listening to device: %s", err)
r.Log(ErrorLog, "Retrying in 1 minute.")
time.Sleep(time.Minute)
continue
}
r.Log(InfoLog, "Connected to input device: %s", r.Device)
// Update stop function for disconnects.
r.ListenerStop = stop
break
}
}()
}
// Try finding input port.
log.Println("Connecting to device:", r.Device)
in, err := midi.FindInPort(r.Device)
if err != nil {
log.Println("Can't find device:", r.Device)
return
}
// Start listening to MIDI messages.
stop, err := midi.ListenTo(in, func(msg midi.Message, timestampms int32) {
var channel, note, velocity uint8
switch {
// Get notes with an velocity set.
case msg.GetNoteStart(&channel, &note, &velocity):
r.Log(ReceiveLog, "starting note %s(%d) on channel %v with velocity %v", midi.Note(note), note, channel, velocity)
// Process request.
r.sendRequest(channel, note, velocity)
// If no velocity is set, an note end message is received.
case msg.GetNoteEnd(&channel, &note):
r.Log(ReceiveLog, "ending note %s(%d) on channel %v", midi.Note(note), note, channel)
// Process request.
r.sendRequest(channel, note, 0)
default:
// ignore
}
})
if err != nil {
r.Log(ErrorLog, "Error listening to device: %s", err)
return
}
// Update stop function for disconnects.
r.ListenerStop = stop
if r.MQTT.Host != "" && r.MQTT.Port != 0 {
// Connect to MQTT.
mqtt_opts := mqtt.NewClientOptions()
mqtt_opts.AddBroker(fmt.Sprintf("tcp://%s:%d", r.MQTT.Host, r.MQTT.Port))
mqtt_opts.SetClientID(r.MQTT.ClientId)
mqtt_opts.SetUsername(r.MQTT.User)
mqtt_opts.SetPassword(r.MQTT.Password)
r.MqttClient = mqtt.NewClient(mqtt_opts)
go func() {
for {
// Connect to MQTT.
mqtt_opts := mqtt.NewClientOptions()
mqtt_opts.AddBroker(fmt.Sprintf("tcp://%s:%d", r.MQTT.Host, r.MQTT.Port))
mqtt_opts.SetClientID(r.MQTT.ClientId)
mqtt_opts.SetUsername(r.MQTT.User)
mqtt_opts.SetPassword(r.MQTT.Password)
r.MqttClient = mqtt.NewClient(mqtt_opts)
// Connect and failures are fatal exiting service.
r.Log(DebugLog, "Connecting to MQTT")
if t := r.MqttClient.Connect(); t.Wait() && t.Error() != nil {
log.Fatalf("MQTT error: %s", t.Error())
return
}
// Connect and failures are fatal exiting service.
r.Log(DebugLog, "Connecting to MQTT")
if t := r.MqttClient.Connect(); t.Wait() && t.Error() != nil {
log.Fatalf("MQTT error: %s", t.Error())
r.Log(ErrorLog, "Retrying in 1 minute.")
time.Sleep(time.Minute)
continue
}
// Subscribe to MQTT topics.
r.MqttSubscribe(r.MQTT.Topic + "/send")
r.MqttSubscribe(r.MQTT.Topic + "/status/check")
// Subscribe to command topics configured.
for _, trig := range r.RequestTriggers {
if trig.MqttTopic != "" {
r.MqttSubscribe(trig.MqttTopic)
// Subscribe to MQTT topics.
r.MqttSubscribe(r.MQTT.Topic + "/send")
r.MqttSubscribe(r.MQTT.Topic + "/status/check")
// Subscribe to command topics configured.
for _, trig := range r.RequestTriggers {
if trig.MqttTopic != "" {
r.MqttSubscribe(trig.MqttTopic)
}
if trig.MqttSubTopic != "" {
r.MqttSubscribe(r.MQTT.Topic + "/" + trig.MqttSubTopic)
}
}
break
}
if trig.MqttSubTopic != "" {
r.MqttSubscribe(r.MQTT.Topic + "/" + trig.MqttSubTopic)
}
}
}()
}
}
@ -539,4 +587,5 @@ func (r *MidiRouter) Disconnect() {
if r.ListenerStop != nil {
r.ListenerStop()
}
r.MqttClient.Disconnect(0)
}