blob: 4af1a3fe08670abece473d785534549c2c07cd81 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package simulated
import (
"context"
"runtime/debug"
"strconv"
"sync"
"time"
"github.com/pkg/errors"
"github.com/rs/zerolog"
apiModel "github.com/apache/plc4x/plc4go/pkg/api/model"
spiModel "github.com/apache/plc4x/plc4go/spi/model"
"github.com/apache/plc4x/plc4go/spi/options"
"github.com/apache/plc4x/plc4go/spi/tracer"
)
type Writer struct {
device *Device
options map[string][]string
tracer tracer.Tracer
wg sync.WaitGroup // use to track spawned go routines
log zerolog.Logger
}
func NewWriter(device *Device, writerOptions map[string][]string, tracer tracer.Tracer, _options ...options.WithOption) *Writer {
customLogger := options.ExtractCustomLoggerOrDefaultToGlobal(_options...)
return &Writer{
device: device,
options: writerOptions,
tracer: tracer,
log: customLogger,
}
}
func (w *Writer) Write(_ context.Context, writeRequest apiModel.PlcWriteRequest) <-chan apiModel.PlcWriteRequestResult {
ch := make(chan apiModel.PlcWriteRequestResult, 1)
w.wg.Go(func() {
defer func() {
if err := recover(); err != nil {
ch <- spiModel.NewDefaultPlcWriteRequestResult(writeRequest, nil, errors.Errorf("panic-ed %v. Stack: %s", err, debug.Stack()))
}
}()
var txId string
if w.tracer != nil {
txId = w.tracer.AddTransactionalStartTrace("write", "started")
}
// Possibly add a delay.
if delayString, ok := w.options["writeDelay"]; ok {
if len(delayString) == 1 {
delay, err := strconv.Atoi(delayString[0])
if err == nil {
time.Sleep(time.Duration(delay) * time.Millisecond)
}
}
}
// Process the request
responseCodes := map[string]apiModel.PlcResponseCode{}
for _, tagName := range writeRequest.GetTagNames() {
tag := writeRequest.GetTag(tagName)
simulatedTagVar, ok := tag.(simulatedTag)
if !ok {
responseCodes[tagName] = apiModel.PlcResponseCode_INVALID_ADDRESS
} else {
plcValue := writeRequest.GetValue(tagName)
w.device.Set(simulatedTagVar, &plcValue)
responseCodes[tagName] = apiModel.PlcResponseCode_OK
}
}
if w.tracer != nil {
w.tracer.AddTransactionalTrace(txId, "write", "success")
}
// Emit the response
ch <- spiModel.NewDefaultPlcWriteRequestResult(writeRequest, spiModel.NewDefaultPlcWriteResponse(writeRequest, responseCodes), nil)
})
return ch
}