| /* |
| * 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 |
| } |