Poller split custom number #23
Replies: 2 comments 20 replies
-
Hi, I'll check tomorrow what can be done. I was planning to make splitting more configurable/extendable. At the moment there is logic that you can use maybe as a workaround. If you add specific query parameters to
If you want to know how it exactly works the place to look is Lines 184 to 193 in 77dd4be |
Beta Was this translation helpful? Give feedback.
-
This will use single connection per address This code creates special shared/single connection function for Used as: config := poller.Config{
ConnectFunc: NewSingleConnectionPerAddressClientFunc(),
}
p := poller.NewPollerWithConfig(batches, config) type SingleConnectionPerAddress struct {
mu sync.Mutex
clientPool map[string]*sharedClient
}
func NewSingleConnectionPerAddressClientFunc() func(ctx context.Context, protocol modbus.ProtocolType, address string) (poller.Client, error) {
pool := &SingleConnectionPerAddress{
clientPool: map[string]*sharedClient{},
}
return pool.NewClientFunc
}
func (scp *SingleConnectionPerAddress) NewClientFunc(ctx context.Context, protocol modbus.ProtocolType, address string) (poller.Client, error) {
scp.mu.Lock()
defer scp.mu.Unlock()
// address could have params `tcp://192.168.1.2:502??invalid_addr=70`
if i := strings.IndexByte(address, '?'); i != -1 {
address = address[:i]
}
client, ok := scp.clientPool[address]
if ok {
return client, nil
}
orgClient, err := poller.DefaultConnectClient(ctx, protocol, address)
if err != nil {
return nil, err
}
client = &sharedClient{
mu: sync.Mutex{},
client: orgClient,
address: address,
onClose: scp.onClose,
}
scp.clientPool[address] = client
return client, nil
}
func (scp *SingleConnectionPerAddress) onClose(address string) {
scp.mu.Lock()
defer scp.mu.Unlock()
delete(scp.clientPool, address)
}
type sharedClient struct {
mu sync.Mutex
client poller.Client
address string
onClose func(address string)
}
func (c *sharedClient) Do(ctx context.Context, req packet.Request) (packet.Response, error) {
c.mu.Lock()
defer c.mu.Unlock()
if c.client == nil {
return nil, modbus.ErrClientNotConnected
}
return c.client.Do(ctx, req)
}
func (c *sharedClient) Close() error {
c.mu.Lock()
defer c.mu.Unlock()
if c.client == nil {
return nil
}
err := c.client.Close()
c.client = nil
c.onClose(c.address)
return err
} |
Beta Was this translation helpful? Give feedback.
-
Hello.
I am testing poller function and have problem with splitting registers.
My device can answer only for the 16 registers.
https://datakom.com.tr/upload/Files/500_MODBUS.pdf
The maximum number of registers read in one message is 16. If more registers are requested, the unit will send only the first 16 registers.
I need to read 29. Can you add custom Split?
For example Split(16)
With this way i got error -
Beta Was this translation helpful? Give feedback.
All reactions