2024-12-15 13:45:46 +01:00
package hardware
import (
"bufio"
2024-12-29 21:22:53 +01:00
"context"
2024-12-15 13:45:46 +01:00
_ "embed"
"fmt"
"io"
2024-12-29 13:09:46 +01:00
"github.com/rs/zerolog/log"
2024-12-15 13:45:46 +01:00
"os"
"os/exec"
)
const (
activateCommandString = 0x01
deactivateCommandString = 0x02
setCommandString = 0x03
)
// FTDIPeripheral contains the data of an FTDI peripheral
type FTDIPeripheral struct {
name string // The name of the peripheral
serialNumber string // The S/N of the FTDI peripheral
location int // The location of the peripheral
universesNumber int // The number of DMX universes handled by this peripheral
programName string // The temp file name of the executable
dmxSender * exec . Cmd // The command to pilot the DMX sender program
stdin io . WriteCloser // For writing in the DMX sender
stdout io . ReadCloser // For reading from the DMX sender
stderr io . ReadCloser // For reading the errors
disconnectChan chan struct { } // Channel to cancel the connection
errorsChan chan error // Channel to get the errors
}
2024-12-29 21:22:53 +01:00
//go:embed third-party/ftdi/dmxSender.exe
var dmxSender [ ] byte
2024-12-15 13:45:46 +01:00
// NewFTDIPeripheral creates a new FTDI peripheral
2024-12-23 17:22:37 +01:00
func NewFTDIPeripheral ( name string , serialNumber string , location int ) ( * FTDIPeripheral , error ) {
2024-12-29 13:09:46 +01:00
log . Info ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "name" , name ) . Str ( "s/n" , serialNumber ) . Int ( "location" , location ) . Msg ( "FTDI peripheral created" )
2024-12-15 13:45:46 +01:00
// Create a temporary file
2024-12-29 21:22:53 +01:00
tempFile , err := os . Create ( fmt . Sprintf ( "dmxSender-%s.exe" , serialNumber ) )
2024-12-15 13:45:46 +01:00
if err != nil {
return nil , err
}
2024-12-29 13:09:46 +01:00
log . Trace ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , serialNumber ) . Msg ( "FTDI sender temp created" )
2024-12-15 13:45:46 +01:00
// Write the embedded executable to the temp file
if _ , err := tempFile . Write ( dmxSender ) ; err != nil {
return nil , err
}
2024-12-29 13:09:46 +01:00
log . Trace ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , serialNumber ) . Msg ( "FTDI sender written" )
2024-12-15 13:45:46 +01:00
tempFile . Close ( )
return & FTDIPeripheral {
name : name ,
2024-12-29 21:22:53 +01:00
dmxSender : nil ,
2024-12-15 13:45:46 +01:00
programName : tempFile . Name ( ) ,
serialNumber : serialNumber ,
location : location ,
universesNumber : 1 ,
disconnectChan : make ( chan struct { } ) ,
errorsChan : make ( chan error , 1 ) ,
} , nil
}
// Connect connects the FTDI peripheral
2024-12-29 21:22:53 +01:00
func ( p * FTDIPeripheral ) Connect ( ctx context . Context ) error {
2024-12-15 13:45:46 +01:00
// Connect if no connection is already running
2024-12-29 13:09:46 +01:00
log . Trace ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "connecting FTDI peripheral..." )
2024-12-15 13:45:46 +01:00
2024-12-29 21:22:53 +01:00
// Check if the connection has already been established
if p . dmxSender != nil {
log . Warn ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "dmxSender already initialized" )
return nil
}
2024-12-15 13:45:46 +01:00
2024-12-29 21:22:53 +01:00
// Initialize the exec.Command for running the process
p . dmxSender = exec . Command ( p . programName , fmt . Sprintf ( "%d" , p . location ) )
log . Trace ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "dmxSender instance created" )
2024-12-15 13:45:46 +01:00
2024-12-29 21:22:53 +01:00
// Create the pipes for stdin, stdout, and stderr asynchronously without blocking
var err error
if p . stdout , err = p . dmxSender . StdoutPipe ( ) ; err != nil {
log . Err ( err ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "unable to create stdout pipe" )
return fmt . Errorf ( "unable to create stdout pipe: %v" , err )
}
if p . stdin , err = p . dmxSender . StdinPipe ( ) ; err != nil {
log . Err ( err ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "unable to create stdin pipe" )
return fmt . Errorf ( "unable to create stdin pipe: %v" , err )
}
if p . stderr , err = p . dmxSender . StderrPipe ( ) ; err != nil {
log . Err ( err ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "unable to create stderr pipe" )
return fmt . Errorf ( "unable to create stderr pipe: %v" , err )
}
2024-12-15 13:45:46 +01:00
2024-12-29 21:22:53 +01:00
// Launch a goroutine to read stderr asynchronously
go func ( ) {
scanner := bufio . NewScanner ( p . stderr )
for scanner . Scan ( ) {
// Process each line read from stderr
log . Err ( fmt . Errorf ( scanner . Text ( ) ) ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "error detected in dmx sender" )
}
if err := scanner . Err ( ) ; err != nil {
log . Err ( err ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "error reading from stderr" )
}
} ( )
// Launch the command asynchronously in another goroutine
go func ( ) {
// Run the command, respecting the context cancellation
err := p . dmxSender . Run ( )
select {
case <- ctx . Done ( ) :
// If the context is canceled, handle it gracefully
log . Warn ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "dmxSender was canceled by context" )
return
default :
// Handle command exit normally
2024-12-15 13:45:46 +01:00
if err != nil {
2024-12-29 21:22:53 +01:00
log . Err ( err ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "error while execution of dmx sender" )
2024-12-15 13:45:46 +01:00
if exitError , ok := err . ( * exec . ExitError ) ; ok {
2024-12-29 13:09:46 +01:00
log . Warn ( ) . Str ( "file" , "FTDIPeripheral" ) . Int ( "exitCode" , exitError . ExitCode ( ) ) . Str ( "s/n" , p . serialNumber ) . Msg ( "dmx sender exited with code" )
2024-12-15 13:45:46 +01:00
}
} else {
2024-12-29 13:09:46 +01:00
log . Debug ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "dmx sender exited successfully" )
2024-12-15 13:45:46 +01:00
}
2024-12-29 21:22:53 +01:00
}
} ( )
log . Debug ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "dmxSender process started successfully" )
2024-12-15 13:45:46 +01:00
return nil
}
// Disconnect disconnects the FTDI peripheral
2024-12-29 21:22:53 +01:00
func ( p * FTDIPeripheral ) Disconnect ( ctx context . Context ) error {
2024-12-29 13:09:46 +01:00
log . Trace ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "disconnecting FTDI peripheral..." )
2024-12-15 13:45:46 +01:00
if p . dmxSender != nil {
2024-12-29 13:09:46 +01:00
log . Debug ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "dmxsender is defined for this FTDI" )
2024-12-15 13:45:46 +01:00
_ , err := io . WriteString ( p . stdin , string ( [ ] byte { 0x04 , 0x00 , 0x00 , 0x00 } ) )
if err != nil {
2024-12-29 13:09:46 +01:00
log . Err ( err ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "unable to write command to sender" )
return fmt . Errorf ( "unable to disconnect: %v" , err )
2024-12-15 13:45:46 +01:00
}
p . stdin . Close ( )
p . stdout . Close ( )
p . dmxSender = nil
err = os . Remove ( p . programName )
if err != nil {
2024-12-29 21:22:53 +01:00
log . Warn ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Str ( "senderPath" , p . programName ) . Msg ( "unable to delete the dmx sender temporary file" )
2024-12-29 13:09:46 +01:00
return fmt . Errorf ( "unable to delete the temporary file: %v" , err )
2024-12-15 13:45:46 +01:00
}
return nil
}
2024-12-29 13:09:46 +01:00
log . Warn ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "error while disconnecting: not connected" )
return fmt . Errorf ( "unable to disconnect: not connected" )
2024-12-15 13:45:46 +01:00
}
// Activate activates the FTDI peripheral
2024-12-29 21:22:53 +01:00
func ( p * FTDIPeripheral ) Activate ( ctx context . Context ) error {
2024-12-15 13:45:46 +01:00
if p . dmxSender != nil {
2024-12-29 13:09:46 +01:00
log . Debug ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "dmxsender is defined for this FTDI" )
2024-12-15 13:45:46 +01:00
_ , err := io . WriteString ( p . stdin , string ( [ ] byte { 0x01 , 0x00 , 0x00 , 0x00 } ) )
if err != nil {
2024-12-29 13:09:46 +01:00
log . Err ( err ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "unable to write command to sender" )
return fmt . Errorf ( "unable to activate: %v" , err )
2024-12-15 13:45:46 +01:00
}
return nil
}
2024-12-29 13:09:46 +01:00
log . Warn ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "error while activating: not connected" )
return fmt . Errorf ( "unable to activate: not connected" )
2024-12-15 13:45:46 +01:00
}
// Deactivate deactivates the FTDI peripheral
2024-12-29 21:22:53 +01:00
func ( p * FTDIPeripheral ) Deactivate ( ctx context . Context ) error {
2024-12-15 13:45:46 +01:00
if p . dmxSender != nil {
2024-12-29 13:09:46 +01:00
log . Debug ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "dmxsender is defined for this FTDI" )
2024-12-15 13:45:46 +01:00
_ , err := io . WriteString ( p . stdin , string ( [ ] byte { 0x02 , 0x00 , 0x00 , 0x00 } ) )
if err != nil {
2024-12-29 13:09:46 +01:00
log . Err ( err ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "unable to write command to sender" )
return fmt . Errorf ( "unable to deactivate: %v" , err )
2024-12-15 13:45:46 +01:00
}
return nil
}
2024-12-29 13:09:46 +01:00
log . Warn ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "error while deactivating: not connected" )
return fmt . Errorf ( "unable to deactivate: not connected" )
2024-12-15 13:45:46 +01:00
}
// SetDeviceProperty sends a command to the specified device
2024-12-29 21:22:53 +01:00
func ( p * FTDIPeripheral ) SetDeviceProperty ( ctx context . Context , uint32 , channelNumber uint32 , channelValue byte ) error {
2024-12-15 13:45:46 +01:00
if p . dmxSender != nil {
2024-12-29 13:09:46 +01:00
log . Debug ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "dmxsender is defined for this FTDI" )
2024-12-15 13:45:46 +01:00
commandString := [ ] byte { 0x03 , 0x01 , 0x00 , 0xff , 0x03 , 0x02 , 0x00 , channelValue }
_ , err := io . WriteString ( p . stdin , string ( commandString ) )
if err != nil {
2024-12-29 13:09:46 +01:00
log . Err ( err ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "unable to write command to sender" )
return fmt . Errorf ( "unable to set device property: %v" , err )
2024-12-15 13:45:46 +01:00
}
return nil
}
2024-12-29 13:09:46 +01:00
log . Warn ( ) . Str ( "file" , "FTDIPeripheral" ) . Str ( "s/n" , p . serialNumber ) . Msg ( "error while setting device property: not connected" )
return fmt . Errorf ( "unable to set device property: not connected" )
2024-12-15 13:45:46 +01:00
}
// GetInfo gets all the peripheral information
func ( p * FTDIPeripheral ) GetInfo ( ) PeripheralInfo {
return PeripheralInfo {
2024-12-20 17:18:57 +01:00
Name : p . name ,
SerialNumber : p . serialNumber ,
ProtocolName : "FTDI" ,
2024-12-15 13:45:46 +01:00
}
}