ttyACM0 포트를 사용할 수 있게 된 후 C#(.NET6)으로 작성된 자체 서비스(SignalR 클라이언트)를 시작해야 합니다. 이 서비스는 다른 서비스 다음에 시작되어야 합니다(SignalR 서버).
정상적으로 재부팅 후 서버 서비스가 시작됩니다(아무 문제 없이). 클라이언트 서비스 상태는 다음 정보를 제공합니다.
● RfidHwSigRClient.service
Loaded: loaded (/etc/systemd/system/RfidHwSigRClient.service;
disabled; vendor preset: enabled) Active: inactive (dead) since Mon
2023-03-06 22:20:58 CET; 2min 17s ago Process: 1583
ExecStart=/usr/bin/dotnet
/home/predator/Projects/VSLinuxDbg/RfidHwSigRClient/RfidHwSigRClient.dll
(code=exited, status=0/SUCCESS) Process: 767 ExecStartPre=/bin/sleep
10 (code=exited, status=0/SUCCESS) Main PID: 1583 (code=exited,
status=0/SUCCESS)
Mar 06 22:20:46 PredatorClient systemd[1]: Starting
RfidHwSigRClient.service... Mar 06 22:20:57 PredatorClient
dotnet[1583]: Microsoft.Hosting.Lifetime[0] Application started.
Hosting environment: Production; Content root path: / Mar 06 22:20:57
PredatorClient systemd[1]: Started RfidHwSigRClient.service. Mar 06
22:20:58 PredatorClient dotnet[1583]:
Microsoft.Extensions.Hosting.Internal.Host[9] BackgroundService failed
System.UnauthorizedAccessException: Access to the port '/dev/usb_rfid'
is denied. ---> System.IO.IOException: Device or resource busy ---
End of inner exception stack trace --- at
System.IO.Ports.SafeSerialDeviceHandle.Open(String portName) at
System.IO.Ports.SerialStream..ctor(String portName, Int32 baudRate,
Parity parity, Int32 dataBits, StopBits stopBits, Int32 readTimeout,
Int32 writeTimeout, Handshake handshake, Boolean dtrEnable, Boolean
rtsEnable, Boolean discardNull, Byte parityReplace) at
System.IO.Ports.SerialPort.Open() at
Predator.Hardware.RfIDCard.RfId.Connect() in
D:\Projects\PredatorDotNet\Source\PredatorClientDotNet\Hardware\RFIDCard\RfId.cs:line
159 at
Predator.SignalRComm.RfidHwSigRClient.RfidBroker.ConnectAsync() in
D:\Projects\PredatorDotNet\Source\PredatorClientDotNet\SignalRComm\Derived\RfidHwSigRClient\RfidHwSigRClient\RfidBroker.cs:line
95 at
Predator.SignalRComm.RfidHwSigRClient.RfidBroker.ExecuteAsync(CancellationToken
stoppingToken) in
D:\Projects\PredatorDotNet\Source\PredatorClientDotNet\SignalRComm\Derived\RfidHwSigRClient\RfidHwSigRClient\RfidBroker.cs:line
77 at
Microsoft.Extensions.Hosting.Internal.Host.TryExecuteBackgroundServiceAsync(BackgroundService
backgroundService) Mar 06 22:20:58 PredatorClient dotnet[1583]:
Microsoft.Extensions.Hosting.Internal.Host[10] The
HostOptions.BackgroundServiceExceptionBehavior is configured to
StopHost. A BackgroundService has thrown an unhandled exception, and
the IHost instance is stopping. To avoid this behavior, configure this
to Ignore; however the BackgroundService will not be restarted.
System.UnauthorizedAccessException: Access to the port '/dev/usb_rfid'
is denied. ---> System.IO.IOException: Device or resource busy ---
End of inner exception stack trace --- at
System.IO.Ports.SafeSerialDeviceHandle.Open(String portName) at
System.IO.Ports.SerialStream..ctor(String portName, Int32 baudRate,
Parity parity, Int32 dataBits, StopBits stopBits, Int32 readTimeout,
Int32 writeTimeout, Handshake handshake, Boolean dtrEnable, Boolean
rtsEnable, Boolean discardNull, Byte parityReplace) at
System.IO.Ports.SerialPort.Open() at
Predator.Hardware.RfIDCard.RfId.Connect() in
D:\Projects\PredatorDotNet\Source\PredatorClientDotNet\Hardware\RFIDCard\RfId.cs:line
159 at
Predator.SignalRComm.RfidHwSigRClient.RfidBroker.ConnectAsync() in
D:\Projects\PredatorDotNet\Source\PredatorClientDotNet\SignalRComm\Derived\RfidHwSigRClient\RfidHwSigRClient\RfidBroker.cs:line
95 at
Predator.SignalRComm.RfidHwSigRClient.RfidBroker.ExecuteAsync(CancellationToken
stoppingToken) in
D:\Projects\PredatorDotNet\Source\PredatorClientDotNet\SignalRComm\Derived\RfidHwSigRClient\RfidHwSigRClient\RfidBroker.cs:line
77 at
Microsoft.Extensions.Hosting.Internal.Host.TryExecuteBackgroundServiceAsync(BackgroundService
backgroundService) Mar 06 22:20:58 PredatorClient dotnet[1583]:
Microsoft.Hosting.Lifetime[0] Application is shutting down...
문제는 포트를 사용할 수 없다는 것입니다.
Ubuntu가 시작되자마자 systemctl restart service_name
명령줄에서 클라이언트 서비스( )를 다시 시작할 수 있으며 서비스가 제대로 작동합니다(포트가 준비되었습니다).
서비스 시작 방법에 관한 여러 포럼을 읽었습니다. 불행히도 성공하지 못했습니다.
내 udev 규칙은 다음과 같습니다.
KERNEL=="ttyACM*", ATTRS{manufacturer}=="MOD elektronik*", ATTRS{product}=="MOD RFID reader", MODE="0666", SYMLINK+="usb_rfid", ENV{SYSTEMD_WANTS}+="RfidHwSigRClient.service"
내 RfidHwSigRClient.service는 다음과 같습니다.
[unit]
Description=RFId Card Reader SignalR Client Service
BindsTo=dev-usb_rfid.device
Requires=MainSigRServer.service
After=dev-usb_rfid.device MainSigRserver.service
StartLimitIntervalSec=300
StartLimitBurst=30
[Service]
User=predator
Group=predator
Type=notify
ExecStartPre=/bin/sleep 10
ExecStart=/usr/bin/dotnet /home/predator/Projects/VSLinuxDbg/RfidHwSigRClient/RfidHwSigRClient.dll
TimeoutStartSec=30s
Restart=on-failure
RestartSec=5s
[Install]
WantedBy=multi-use
또한 10초 후에 서비스가 시작되는지 확인했습니다( ExecStartPre=/bin/sleep 10
). 성공하지 못했습니다. 사용자 포식자는 그룹 다이얼아웃에 속합니다. 우분투는 18.04 LTS입니다.
제발, 내가 뭘 잘못하고 있는지 아는 사람 있나요? 감사합니다. 아르 자형.
답변1
나는 대략적으로 수면을 취해야했습니다. 15초 동안 클라이언트 서비스를 시작합니다. 수면이 10초라면. 포트가 준비되지 않았습니다(액세스가 거부되었습니다). 그러나 포트와 서버 서비스가 모두 준비되자마자 클라이언트 서비스를 시작하는 방법은 아직 모르겠습니다.
내 온도. RfidHwSigRClient.service의 구성은 다음과 같습니다.
[Unit]
Description=RFId Card Reader SignalR Client Service
BindsTo=dev-usb_rfid.device
After=dev-usb_rfid.device MainSigRServer.service
StartLimitIntervalSec=300
StartLimitBurst=30
[Service]
Type=notify
ExecStartPre=/bin/sleep 15
ExecStart=/usr/bin/dotnet /home/predator/Projects/VSLinuxDbg/RfidHwSigRClient/RfidHwSigRClient.dll
Restart=on-failure
RestartSec=5s
User=predator
Group=predator
[Install]
WantedBy=dev-usb_rfid.device
Udev 규칙 99-usb-serial.rules는 다음과 같습니다.
KERNEL=="ttyACM*", ATTRS{manufacturer}=="MOD elektronik*", ATTRS{product}=="MOD RFID reader", SYMLINK+="usb_rfid", MODE="0666", ACTION=="add", RUN+="/home/predator/Projects/Scripts/rfid_plugged.sh", TAG+="systemd" ENV{SYSTEMD_WANTS}+="MainSigRServer.service"
감사합니다..