Compare commits
	
		
			801 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					99eef8fb28 | ||
| 
						 | 
					47e7685d39 | ||
| 
						 | 
					b6ea596ade | ||
| 
						 | 
					44d60b469b | ||
| 
						 | 
					51087408df | ||
| 
						 | 
					96226d9e6e | ||
| 
						 | 
					28f0f62424 | ||
| 
						 | 
					2e772a8cd4 | ||
| 
						 | 
					3fd192f0cf | ||
| 
						 | 
					d16ae81961 | ||
| 
						 | 
					211a8e093e | ||
| 
						 | 
					84010a51d6 | ||
| 
						 | 
					e9700ea19b | ||
| 
						 | 
					e81ec68072 | ||
| 
						 | 
					6096e97708 | ||
| 
						 | 
					6f302b66b4 | ||
| 
						 | 
					b6e41890f7 | ||
| 
						 | 
					be5332a048 | ||
| 
						 | 
					4d72aeb0cd | ||
| 
						 | 
					c7c3c35100 | ||
| 
						 | 
					2f0df3c552 | ||
| 
						 | 
					e6effcd921 | ||
| 
						 | 
					cd462b9523 | ||
| 
						 | 
					7e2e17d38b | ||
| 
						 | 
					665d616a06 | ||
| 
						 | 
					ced90f5bb9 | ||
| 
						 | 
					ca0ddb6cb4 | ||
| 
						 | 
					3d733813e1 | ||
| 
						 | 
					45ddc0b154 | ||
| 
						 | 
					87294497f3 | ||
| 
						 | 
					5f45f9c0d0 | ||
| 
						 | 
					7f78c06793 | ||
| 
						 | 
					8ea5b7eebc | ||
| 
						 | 
					e49cd511af | ||
| 
						 | 
					15c88646df | ||
| 
						 | 
					f47919543d | ||
| 
						 | 
					a95291d9cd | ||
| 
						 | 
					033deb3d29 | ||
| 
						 | 
					923994c1f4 | ||
| 
						 | 
					3d45b839b4 | ||
| 
						 | 
					55d1efa212 | ||
| 
						 | 
					21d794a0e5 | ||
| 
						 | 
					614022a78c | ||
| 
						 | 
					c00688e23a | ||
| 
						 | 
					5c69917d19 | ||
| 
						 | 
					6a5eb75b6c | ||
| 
						 | 
					727c55eaa2 | ||
| 
						 | 
					9b03173ec5 | ||
| 
						 | 
					ab95855d6d | ||
| 
						 | 
					bb84594c6b | ||
| 
						 | 
					97392c76b1 | ||
| 
						 | 
					53aec2b306 | ||
| 
						 | 
					a0a381dc63 | ||
| 
						 | 
					4fa95edeec | ||
| 
						 | 
					b944f1d70e | ||
| 
						 | 
					0f1a5d0085 | ||
| 
						 | 
					e7e10222e7 | ||
| 
						 | 
					87c4fda588 | ||
| 
						 | 
					0ff820de6f | ||
| 
						 | 
					0dcf55e6d9 | ||
| 
						 | 
					a69842c910 | ||
| 
						 | 
					318f635e4d | ||
| 
						 | 
					abf0d72316 | ||
| 
						 | 
					5cd89c8844 | ||
| 
						 | 
					501c1af38c | ||
| 
						 | 
					8526e130d9 | ||
| 
						 | 
					76fd08eade | ||
| 
						 | 
					bfe48ae9d2 | ||
| 
						 | 
					8547cdba7c | ||
| 
						 | 
					00d0cd631d | ||
| 
						 | 
					04460d13cd | ||
| 
						 | 
					a17a15e5d7 | ||
| 
						 | 
					bfe2465658 | ||
| 
						 | 
					4568987785 | ||
| 
						 | 
					c0e5a1419d | ||
| 
						 | 
					23ac7ab748 | ||
| 
						 | 
					5622dc74a8 | ||
| 
						 | 
					528986f4f0 | ||
| 
						 | 
					e6b8ff3f91 | ||
| 
						 | 
					3fc3c5296f | ||
| 
						 | 
					f4f2a14a31 | ||
| 
						 | 
					fa3f464cb6 | ||
| 
						 | 
					bed20538ae | ||
| 
						 | 
					3da461c6e6 | ||
| 
						 | 
					9c73e8452a | ||
| 
						 | 
					750dab69d3 | ||
| 
						 | 
					702d0da47c | ||
| 
						 | 
					3a955b1e53 | ||
| 
						 | 
					d408bd93e8 | ||
| 
						 | 
					63fd9eb1de | ||
| 
						 | 
					4718ab6ddf | ||
| 
						 | 
					affd4b3a26 | ||
| 
						 | 
					6a660dbcda | ||
| 
						 | 
					5ff3af8b48 | ||
| 
						 | 
					9d49070f93 | ||
| 
						 | 
					5b3994a1dd | ||
| 
						 | 
					bb1e62c580 | ||
| 
						 | 
					29cc776908 | ||
| 
						 | 
					d2ef60b2ec | ||
| 
						 | 
					41ea496d41 | ||
| 
						 | 
					c69bb38929 | ||
| 
						 | 
					23b3f8494e | ||
| 
						 | 
					0ff94eec1b | ||
| 
						 | 
					f67a638565 | ||
| 
						 | 
					778cf71c61 | ||
| 
						 | 
					e2da083b48 | ||
| 
						 | 
					dcc48c0d4b | ||
| 
						 | 
					2446b4a591 | ||
| 
						 | 
					a973d633c8 | ||
| 
						 | 
					c3b11e6e0f | ||
| 
						 | 
					44bec2d99e | ||
| 
						 | 
					0d59d0a3f6 | ||
| 
						 | 
					ddc8372b09 | ||
| 
						 | 
					ea7325f4e0 | ||
| 
						 | 
					6048f95415 | ||
| 
						 | 
					5682705b8d | ||
| 
						 | 
					6c3b82607d | ||
| 
						 | 
					486c9f4ebf | ||
| 
						 | 
					7c8adebf2d | ||
| 
						 | 
					ce98bb0dc1 | ||
| 
						 | 
					60b49a4296 | ||
| 
						 | 
					5d0a2ffae0 | ||
| 
						 | 
					bbcf7c722a | ||
| 
						 | 
					66574eec6d | ||
| 
						 | 
					305df3a42c | ||
| 
						 | 
					323c6f4270 | ||
| 
						 | 
					c056b2b14c | ||
| 
						 | 
					02ebb07fd8 | ||
| 
						 | 
					59db0890c3 | ||
| 
						 | 
					3ea0d9ed3a | ||
| 
						 | 
					0371d11dd5 | ||
| 
						 | 
					12f1ebaee9 | ||
| 
						 | 
					5202cc22d1 | ||
| 
						 | 
					07a34bdcbf | ||
| 
						 | 
					27806da4be | ||
| 
						 | 
					a15dd1a3ce | ||
| 
						 | 
					6cc7451c7c | ||
| 
						 | 
					652aeee782 | ||
| 
						 | 
					8e1e887b8f | ||
| 
						 | 
					6601a2de64 | ||
| 
						 | 
					75a42ffc32 | ||
| 
						 | 
					8a80a52f5f | ||
| 
						 | 
					b373a434ba | ||
| 
						 | 
					cd78fdafe6 | ||
| 
						 | 
					7a1b4c6db7 | ||
| 
						 | 
					f2c7664033 | ||
| 
						 | 
					12b295f067 | ||
| 
						 | 
					3ee46da40f | ||
| 
						 | 
					beadaa7212 | ||
| 
						 | 
					16ef09426f | ||
| 
						 | 
					b8c1f1f5a9 | ||
| 
						 | 
					e027b5cbd6 | ||
| 
						 | 
					bf1bd73ad1 | ||
| 
						 | 
					86fc95119a | ||
| 
						 | 
					b3a76ea17b | ||
| 
						 | 
					bababd9d53 | ||
| 
						 | 
					8cd5180531 | ||
| 
						 | 
					f1ccbade8c | ||
| 
						 | 
					a66f4b0417 | ||
| 
						 | 
					ff495b2261 | ||
| 
						 | 
					90e5824212 | ||
| 
						 | 
					3d64021062 | ||
| 
						 | 
					fb8ed7428d | ||
| 
						 | 
					bdc273c10a | ||
| 
						 | 
					2d96cffdc7 | ||
| 
						 | 
					4356fccbcd | ||
| 
						 | 
					a169fd4ce7 | ||
| 
						 | 
					dcd418139e | ||
| 
						 | 
					0cac2062d3 | ||
| 
						 | 
					854c5d4ade | ||
| 
						 | 
					0f96c6ec84 | ||
| 
						 | 
					b219bd66c1 | ||
| 
						 | 
					cb52f2c0b3 | ||
| 
						 | 
					d3273e03ef | ||
| 
						 | 
					afbfd963f3 | ||
| 
						 | 
					2cd101c5f3 | ||
| 
						 | 
					b10bc24dee | ||
| 
						 | 
					51a23df861 | ||
| 
						 | 
					03a0a9dad9 | ||
| 
						 | 
					362c0affbe | ||
| 
						 | 
					d5b523479f | ||
| 
						 | 
					7719b8f6d7 | ||
| 
						 | 
					3656c0d524 | ||
| 
						 | 
					8b9e0dd6ea | ||
| 
						 | 
					b3921b1037 | ||
| 
						 | 
					d7d96e5dbf | ||
| 
						 | 
					3934395b7b | ||
| 
						 | 
					04fbe9a529 | ||
| 
						 | 
					badd6dd9a2 | ||
| 
						 | 
					059082456d | ||
| 
						 | 
					405b68e22f | ||
| 
						 | 
					c90bf6692c | ||
| 
						 | 
					b5ea5d0cc5 | ||
| 
						 | 
					243617a1e1 | ||
| 
						 | 
					9e703edd59 | ||
| 
						 | 
					d6c2cf2810 | ||
| 
						 | 
					ae7811acfa | ||
| 
						 | 
					e55731c099 | ||
| 
						 | 
					9df5c74da4 | ||
| 
						 | 
					8b75f9f785 | ||
| 
						 | 
					bf12428cf4 | ||
| 
						 | 
					b551df978b | ||
| 
						 | 
					c91d85d2f0 | ||
| 
						 | 
					965237fa1f | ||
| 
						 | 
					24cd9afc06 | ||
| 
						 | 
					602fdefebd | ||
| 
						 | 
					7e7818aa17 | ||
| 
						 | 
					27a6023a6a | ||
| 
						 | 
					3861879900 | ||
| 
						 | 
					807eeb8351 | ||
| 
						 | 
					045fa53d58 | ||
| 
						 | 
					69d405ece7 | ||
| 
						 | 
					d3dc4d0c5b | ||
| 
						 | 
					7beba1188e | ||
| 
						 | 
					9779ebe12c | ||
| 
						 | 
					f72cfaa093 | ||
| 
						 | 
					cf8ccafb4a | ||
| 
						 | 
					7fe281b0b8 | ||
| 
						 | 
					5f70e9574f | ||
| 
						 | 
					4be51923ba | ||
| 
						 | 
					a72c78f4a4 | ||
| 
						 | 
					b4d32a6de1 | ||
| 
						 | 
					8a6b2d5daa | ||
| 
						 | 
					bebadeef14 | ||
| 
						 | 
					3551be67f0 | ||
| 
						 | 
					1103950b25 | ||
| 
						 | 
					9d06e01cec | ||
| 
						 | 
					24be946b7e | ||
| 
						 | 
					f56d2fc60f | ||
| 
						 | 
					9f8356f409 | ||
| 
						 | 
					8e6345d938 | ||
| 
						 | 
					0eac78bd1c | ||
| 
						 | 
					21db2502d0 | ||
| 
						 | 
					38c3492123 | ||
| 
						 | 
					6376bff476 | ||
| 
						 | 
					38495a3ebc | ||
| 
						 | 
					cf00897be2 | ||
| 
						 | 
					da640f24ec | ||
| 
						 | 
					ed6a777c42 | ||
| 
						 | 
					81af4485d3 | ||
| 
						 | 
					94d8f0237d | ||
| 
						 | 
					c5e579dd38 | ||
| 
						 | 
					7865e76ee2 | ||
| 
						 | 
					4d37212cc0 | ||
| 
						 | 
					9ca43f0bb4 | ||
| 
						 | 
					fe8d262175 | ||
| 
						 | 
					493d6e7165 | ||
| 
						 | 
					17b630bcd8 | ||
| 
						 | 
					018c5ad3b4 | ||
| 
						 | 
					a14b4da977 | ||
| 
						 | 
					01b4597cc7 | ||
| 
						 | 
					0b083ccc1c | ||
| 
						 | 
					ca22cb0eae | ||
| 
						 | 
					bcbb8e5f8c | ||
| 
						 | 
					bea286cdd4 | ||
| 
						 | 
					892d2ee8d2 | ||
| 
						 | 
					7440608c14 | ||
| 
						 | 
					6246f0295b | ||
| 
						 | 
					53c39d9b43 | ||
| 
						 | 
					9a50b60031 | ||
| 
						 | 
					fd74527320 | ||
| 
						 | 
					f91fb522ac | ||
| 
						 | 
					7019c02b18 | ||
| 
						 | 
					94f8422b9b | ||
| 
						 | 
					b25c01567a | ||
| 
						 | 
					b9329885de | ||
| 
						 | 
					ac88c4cff9 | ||
| 
						 | 
					9674f2d238 | ||
| 
						 | 
					38c9ecb6f1 | ||
| 
						 | 
					be9e9cdd5a | ||
| 
						 | 
					afdaf07446 | ||
| 
						 | 
					5caaf10225 | ||
| 
						 | 
					daffc3a776 | ||
| 
						 | 
					34c9748ce5 | ||
| 
						 | 
					f17acbf4d1 | ||
| 
						 | 
					229f56ce2f | ||
| 
						 | 
					0847bedc51 | ||
| 
						 | 
					9e4546c305 | ||
| 
						 | 
					97b01bf26a | ||
| 
						 | 
					e7011628d0 | ||
| 
						 | 
					044ffdecbd | ||
| 
						 | 
					1a06a3543b | ||
| 
						 | 
					5fedc0bf1e | ||
| 
						 | 
					7a881f867e | ||
| 
						 | 
					dfe73a6cfb | ||
| 
						 | 
					45d0fb27ad | ||
| 
						 | 
					31890af61d | ||
| 
						 | 
					a54b3ecc15 | ||
| 
						 | 
					c37cd1bd51 | ||
| 
						 | 
					0e3ee89cbf | ||
| 
						 | 
					a049176c14 | ||
| 
						 | 
					5ef2b0089e | ||
| 
						 | 
					1cc4899593 | ||
| 
						 | 
					26d1390be9 | ||
| 
						 | 
					dc65584fb3 | ||
| 
						 | 
					b3d5a708d5 | ||
| 
						 | 
					904a5d296b | ||
| 
						 | 
					cc82e6a47a | ||
| 
						 | 
					a9d02ec854 | ||
| 
						 | 
					6b5a2d3767 | ||
| 
						 | 
					10d9060b94 | ||
| 
						 | 
					8441839c01 | ||
| 
						 | 
					b669f5838f | ||
| 
						 | 
					1a942f9ce1 | ||
| 
						 | 
					ae840758f7 | ||
| 
						 | 
					082c08e5f9 | ||
| 
						 | 
					74a43557ab | ||
| 
						 | 
					184271789d | ||
| 
						 | 
					fa61b0cad8 | ||
| 
						 | 
					7b197a90d1 | ||
| 
						 | 
					62384af2f6 | ||
| 
						 | 
					b466202a1a | ||
| 
						 | 
					7ab38c18d8 | ||
| 
						 | 
					39841ee43e | ||
| 
						 | 
					e419800d98 | ||
| 
						 | 
					63c99ab69a | ||
| 
						 | 
					e2a8fd2279 | ||
| 
						 | 
					bbde520471 | ||
| 
						 | 
					45fb12f98e | ||
| 
						 | 
					4be3dcce50 | ||
| 
						 | 
					d12a41c769 | ||
| 
						 | 
					ed9a58c9ed | ||
| 
						 | 
					4389cea5a1 | ||
| 
						 | 
					d24854920b | ||
| 
						 | 
					9902443bee | ||
| 
						 | 
					8040b2ef16 | ||
| 
						 | 
					4533680b10 | ||
| 
						 | 
					3b28175135 | ||
| 
						 | 
					8e22812265 | ||
| 
						 | 
					409bc91b9e | ||
| 
						 | 
					61fe543a2a | ||
| 
						 | 
					8a949a7e64 | ||
| 
						 | 
					b0dc9fb97a | ||
| 
						 | 
					83114d1002 | ||
| 
						 | 
					94a25a903f | ||
| 
						 | 
					4f402f9e55 | ||
| 
						 | 
					27bb2e3dcc | ||
| 
						 | 
					66cc52f6ec | ||
| 
						 | 
					ed1367b116 | ||
| 
						 | 
					5c055352e4 | ||
| 
						 | 
					8dcff5ada1 | ||
| 
						 | 
					4b8b23d7d5 | ||
| 
						 | 
					e576f6aed3 | ||
| 
						 | 
					97ab04da91 | ||
| 
						 | 
					c6361bb36e | ||
| 
						 | 
					b0b3dc225d | ||
| 
						 | 
					9cbaba192a | ||
| 
						 | 
					f511598e13 | ||
| 
						 | 
					5ffff3b7aa | ||
| 
						 | 
					24c4fa1c4f | ||
| 
						 | 
					7f312c1273 | ||
| 
						 | 
					ef5c226c81 | ||
| 
						 | 
					3f029cf799 | ||
| 
						 | 
					5bf64a8bec | ||
| 
						 | 
					89f9c537f8 | ||
| 
						 | 
					e8d7f57818 | ||
| 
						 | 
					9b42fa78e7 | ||
| 
						 | 
					96d98e8f39 | ||
| 
						 | 
					1f3a281e78 | ||
| 
						 | 
					dab4b66ee0 | ||
| 
						 | 
					64dd192ddb | ||
| 
						 | 
					b0a5c383d0 | ||
| 
						 | 
					4836cc99eb | ||
| 
						 | 
					ba33873354 | ||
| 
						 | 
					12c838cab0 | ||
| 
						 | 
					a83a1c83a7 | ||
| 
						 | 
					2234f47170 | ||
| 
						 | 
					4bae1b17fc | ||
| 
						 | 
					6a7a0e0fa6 | ||
| 
						 | 
					bff987c55f | ||
| 
						 | 
					e467006913 | ||
| 
						 | 
					1d646d4a6d | ||
| 
						 | 
					70bd4936a1 | ||
| 
						 | 
					5ae7add6d2 | ||
| 
						 | 
					02b206ce7c | ||
| 
						 | 
					2ea626a567 | ||
| 
						 | 
					653909ec11 | ||
| 
						 | 
					1d7c148eaf | ||
| 
						 | 
					94a7e130ca | ||
| 
						 | 
					609abacdf0 | ||
| 
						 | 
					033ff2e49a | ||
| 
						 | 
					00a9bf0641 | ||
| 
						 | 
					100f322456 | ||
| 
						 | 
					2251e08d30 | ||
| 
						 | 
					14e9beef53 | ||
| 
						 | 
					c7cce23d54 | ||
| 
						 | 
					7becf32352 | ||
| 
						 | 
					741eda9f4e | ||
| 
						 | 
					473f51481c | ||
| 
						 | 
					684f9bd11b | ||
| 
						 | 
					b4b7f0c360 | ||
| 
						 | 
					cf42d02628 | ||
| 
						 | 
					cc9a9eab7b | ||
| 
						 | 
					784d6abc9b | ||
| 
						 | 
					0eafc07184 | ||
| 
						 | 
					7e9499bd2e | ||
| 
						 | 
					d01a52aa95 | ||
| 
						 | 
					6bcd492980 | ||
| 
						 | 
					96b15da04b | ||
| 
						 | 
					e0f1801693 | ||
| 
						 | 
					d195ad85c5 | ||
| 
						 | 
					773f7b6c1b | ||
| 
						 | 
					40f29fe399 | ||
| 
						 | 
					b19eb117c1 | ||
| 
						 | 
					f930ba5eff | ||
| 
						 | 
					fbbda6c230 | ||
| 
						 | 
					47db4968e5 | ||
| 
						 | 
					1aff2e518c | ||
| 
						 | 
					24134ca49e | ||
| 
						 | 
					aadbe05a5e | ||
| 
						 | 
					18eddf4700 | ||
| 
						 | 
					0869ba7e70 | ||
| 
						 | 
					cba5db1c53 | ||
| 
						 | 
					0aa5430e23 | ||
| 
						 | 
					88a91828cb | ||
| 
						 | 
					27af89f5fe | ||
| 
						 | 
					6f6d483bfe | ||
| 
						 | 
					915072c191 | ||
| 
						 | 
					76077b92ae | ||
| 
						 | 
					f0d1a527a5 | ||
| 
						 | 
					fdb2a96d1c | ||
| 
						 | 
					849e3ac187 | ||
| 
						 | 
					5c4cf21c59 | ||
| 
						 | 
					40ee3e5d9b | ||
| 
						 | 
					77a7cc034e | ||
| 
						 | 
					82b7881765 | ||
| 
						 | 
					3515775267 | ||
| 
						 | 
					10e12aa1c1 | ||
| 
						 | 
					7fb4e62aa3 | ||
| 
						 | 
					2605eaa614 | ||
| 
						 | 
					8e97b4820d | ||
| 
						 | 
					89af4cdb68 | ||
| 
						 | 
					6099c1a25a | ||
| 
						 | 
					fd23758aea | ||
| 
						 | 
					8ad0b89db1 | ||
| 
						 | 
					e7dffa3ff4 | ||
| 
						 | 
					2772b5d2e2 | ||
| 
						 | 
					df6d9b5f70 | ||
| 
						 | 
					c9622d6d57 | ||
| 
						 | 
					1f0b7c3c7e | ||
| 
						 | 
					c12245037e | ||
| 
						 | 
					fe6d2a50a5 | ||
| 
						 | 
					7be84950f8 | ||
| 
						 | 
					ba212da222 | ||
| 
						 | 
					122b833256 | ||
| 
						 | 
					7c73479041 | ||
| 
						 | 
					151f2e99ae | ||
| 
						 | 
					da367e813f | ||
| 
						 | 
					9143dfe336 | ||
| 
						 | 
					29914d6a72 | ||
| 
						 | 
					0dbef72a97 | ||
| 
						 | 
					3bb118bfe1 | ||
| 
						 | 
					85ce3be077 | ||
| 
						 | 
					064c2bf0a8 | ||
| 
						 | 
					c96d06ccc5 | ||
| 
						 | 
					a658dbb753 | ||
| 
						 | 
					bf5d5b629e | ||
| 
						 | 
					241d576519 | ||
| 
						 | 
					7807e9bdd0 | ||
| 
						 | 
					fcbf15fed6 | ||
| 
						 | 
					cd90a11209 | ||
| 
						 | 
					ff65707ea2 | ||
| 
						 | 
					b16026070c | ||
| 
						 | 
					1b5fbf86d8 | ||
| 
						 | 
					5eb1be8101 | ||
| 
						 | 
					6a863cd26a | ||
| 
						 | 
					87ebb7b6c7 | ||
| 
						 | 
					1233a74307 | ||
| 
						 | 
					221890acfb | ||
| 
						 | 
					687c7e766e | ||
| 
						 | 
					ba22c407d3 | ||
| 
						 | 
					a8b9ed56b5 | ||
| 
						 | 
					3004869e19 | ||
| 
						 | 
					1145853fe1 | ||
| 
						 | 
					f1617a25b1 | ||
| 
						 | 
					932be558d9 | ||
| 
						 | 
					ee3a37d3b9 | ||
| 
						 | 
					5ac412a582 | ||
| 
						 | 
					1625290bc3 | ||
| 
						 | 
					1ec169ad49 | ||
| 
						 | 
					53be552e44 | ||
| 
						 | 
					d3a75d46b9 | ||
| 
						 | 
					256512c961 | ||
| 
						 | 
					5c12ac5bcc | ||
| 
						 | 
					02a2bcb113 | ||
| 
						 | 
					1f0a01c725 | ||
| 
						 | 
					6ea164ede1 | ||
| 
						 | 
					65bae85ecc | ||
| 
						 | 
					2fd7dcf4d0 | ||
| 
						 | 
					aa3bbbe038 | ||
| 
						 | 
					6cb09a6f95 | ||
| 
						 | 
					45868f05d3 | ||
| 
						 | 
					aac7401e20 | ||
| 
						 | 
					c2fc290edd | ||
| 
						 | 
					330357fc36 | ||
| 
						 | 
					f4a3d6a64e | ||
| 
						 | 
					2e0963ec81 | ||
| 
						 | 
					897cb6e62a | ||
| 
						 | 
					80868bd48e | ||
| 
						 | 
					b3df78c56f | ||
| 
						 | 
					783a4259e3 | ||
| 
						 | 
					fcf19b8dc8 | ||
| 
						 | 
					1f9f89817d | ||
| 
						 | 
					7b94da7d85 | ||
| 
						 | 
					164406f6c2 | ||
| 
						 | 
					90ef2adc6b | ||
| 
						 | 
					d65f10f88b | ||
| 
						 | 
					dca0ece9e0 | ||
| 
						 | 
					baabc155c8 | ||
| 
						 | 
					7eb94412d6 | ||
| 
						 | 
					0fc8b24f85 | ||
| 
						 | 
					88f6ef5b96 | ||
| 
						 | 
					7442483419 | ||
| 
						 | 
					9c61933c04 | ||
| 
						 | 
					2b36a99720 | ||
| 
						 | 
					c2cfc42ba4 | ||
| 
						 | 
					4f32704e08 | ||
| 
						 | 
					fccf43685f | ||
| 
						 | 
					39135d81ad | ||
| 
						 | 
					ff4b10681e | ||
| 
						 | 
					10b7908fc2 | ||
| 
						 | 
					31790da8c6 | ||
| 
						 | 
					4621201c47 | ||
| 
						 | 
					692ac31dbf | ||
| 
						 | 
					ed1d163d55 | ||
| 
						 | 
					e931c9040c | ||
| 
						 | 
					f23cb48ea4 | ||
| 
						 | 
					c3b2b6b07b | ||
| 
						 | 
					f89bf590ba | ||
| 
						 | 
					fea17dc00b | ||
| 
						 | 
					dbb1069920 | ||
| 
						 | 
					afc4ccfaa9 | ||
| 
						 | 
					38d55a1c07 | ||
| 
						 | 
					5fca29f103 | ||
| 
						 | 
					78f34b2ca4 | ||
| 
						 | 
					8e8028e809 | ||
| 
						 | 
					8dc70687f8 | ||
| 
						 | 
					fa22f9ee64 | ||
| 
						 | 
					b33c0d3f81 | ||
| 
						 | 
					339009add4 | ||
| 
						 | 
					798c823c4b | ||
| 
						 | 
					14ad86f2a3 | ||
| 
						 | 
					a3a714dc17 | ||
| 
						 | 
					e8d8b0d41d | ||
| 
						 | 
					17daad8f89 | ||
| 
						 | 
					e178263b3b | ||
| 
						 | 
					4febbc261e | ||
| 
						 | 
					ef3a6942fd | ||
| 
						 | 
					f8ac2be62b | ||
| 
						 | 
					25447c34e5 | ||
| 
						 | 
					37cdf8fd48 | ||
| 
						 | 
					ac8e7dc959 | ||
| 
						 | 
					8293382840 | ||
| 
						 | 
					d85a3e7743 | ||
| 
						 | 
					f3f5ffb5c8 | ||
| 
						 | 
					99e6702c62 | ||
| 
						 | 
					e143c25078 | ||
| 
						 | 
					3fb67972be | ||
| 
						 | 
					61c04d4e09 | ||
| 
						 | 
					b22f728958 | ||
| 
						 | 
					112be7e383 | ||
| 
						 | 
					6a7f83fed5 | ||
| 
						 | 
					4b9698a735 | ||
| 
						 | 
					ec3d203a6d | ||
| 
						 | 
					e77b8f5475 | ||
| 
						 | 
					da8bbd321f | ||
| 
						 | 
					98fd011def | ||
| 
						 | 
					9dccbc5316 | ||
| 
						 | 
					03fa26daf9 | ||
| 
						 | 
					fbc41e3895 | ||
| 
						 | 
					f1a6d0c02c | ||
| 
						 | 
					67f6bb7155 | ||
| 
						 | 
					86282f596c | ||
| 
						 | 
					e0f24c795c | ||
| 
						 | 
					7d44ed860c | ||
| 
						 | 
					091094a24c | ||
| 
						 | 
					ba18ee518c | ||
| 
						 | 
					513a031691 | ||
| 
						 | 
					cc79de1106 | ||
| 
						 | 
					bd90e8efb2 | ||
| 
						 | 
					4bd88ff11d | ||
| 
						 | 
					d27ee61292 | ||
| 
						 | 
					0c20b3345f | ||
| 
						 | 
					0f3c8c7193 | ||
| 
						 | 
					16761ec605 | ||
| 
						 | 
					1069440cda | ||
| 
						 | 
					60f5702f17 | ||
| 
						 | 
					ee70133e47 | ||
| 
						 | 
					06a9fdeb2e | ||
| 
						 | 
					f14bd62004 | ||
| 
						 | 
					67cbaf22b7 | ||
| 
						 | 
					60b166dba2 | ||
| 
						 | 
					66e1647ede | ||
| 
						 | 
					a1a35c00a5 | ||
| 
						 | 
					b02e3361c4 | ||
| 
						 | 
					2b9ceaa25a | ||
| 
						 | 
					109b5a9755 | ||
| 
						 | 
					cc4df86c10 | ||
| 
						 | 
					e93532c395 | ||
| 
						 | 
					6ff688326a | ||
| 
						 | 
					eda5d2872f | ||
| 
						 | 
					dc88394f5f | ||
| 
						 | 
					ec85c9a2c6 | ||
| 
						 | 
					1341638556 | ||
| 
						 | 
					4875dfee11 | ||
| 
						 | 
					d834ba8bd4 | ||
| 
						 | 
					6191067771 | ||
| 
						 | 
					6c3a571163 | ||
| 
						 | 
					5efb07e10e | ||
| 
						 | 
					b25ec18ce6 | ||
| 
						 | 
					f3cd281241 | ||
| 
						 | 
					58b93cbf4c | ||
| 
						 | 
					57fc0349ff | ||
| 
						 | 
					d0a2cea772 | ||
| 
						 | 
					58d5801fb5 | ||
| 
						 | 
					e5f2e59798 | ||
| 
						 | 
					1166921057 | ||
| 
						 | 
					aeb49576f2 | ||
| 
						 | 
					7ef1fecef8 | ||
| 
						 | 
					fd630373b5 | ||
| 
						 | 
					d365d8f170 | ||
| 
						 | 
					dfa5e1172f | ||
| 
						 | 
					daf195898a | ||
| 
						 | 
					c61c0a39cf | ||
| 
						 | 
					6137e6baa5 | ||
| 
						 | 
					fd16fd9ffe | ||
| 
						 | 
					2b8bd5f2cc | ||
| 
						 | 
					d312c2e9e7 | ||
| 
						 | 
					578b0d2268 | ||
| 
						 | 
					ffb41b0109 | ||
| 
						 | 
					c63fb5d796 | ||
| 
						 | 
					8caee732e8 | ||
| 
						 | 
					df5381adce | ||
| 
						 | 
					f34836b7fa | ||
| 
						 | 
					44b459883a | ||
| 
						 | 
					14a8592ae3 | ||
| 
						 | 
					787d0dce4a | ||
| 
						 | 
					8be05ff93d | ||
| 
						 | 
					3014af565c | ||
| 
						 | 
					315252bdc4 | ||
| 
						 | 
					c36c3b4607 | ||
| 
						 | 
					7d5e939bab | ||
| 
						 | 
					f75c9d1eed | ||
| 
						 | 
					38cb9855ea | ||
| 
						 | 
					ed8b56d624 | ||
| 
						 | 
					bca48b13ae | ||
| 
						 | 
					29426edb05 | ||
| 
						 | 
					33a2dc687f | ||
| 
						 | 
					e2398a21b2 | ||
| 
						 | 
					f19530276e | ||
| 
						 | 
					2b9b92a78c | ||
| 
						 | 
					99c55dac10 | ||
| 
						 | 
					25667e46f9 | ||
| 
						 | 
					2f4e8f2399 | ||
| 
						 | 
					9169183769 | ||
| 
						 | 
					c420f50831 | ||
| 
						 | 
					c6c9279ef4 | ||
| 
						 | 
					c125e2991d | ||
| 
						 | 
					5348b19d6a | ||
| 
						 | 
					afd426daac | ||
| 
						 | 
					202c511cfa | ||
| 
						 | 
					651eb295a4 | ||
| 
						 | 
					d798aaed33 | ||
| 
						 | 
					63ef347cc9 | ||
| 
						 | 
					3139b2d5a0 | ||
| 
						 | 
					2c80bbb244 | ||
| 
						 | 
					af06755ada | ||
| 
						 | 
					97d6dbaa6c | ||
| 
						 | 
					74cf82a1c7 | ||
| 
						 | 
					74634889ab | ||
| 
						 | 
					dbe30fcd77 | ||
| 
						 | 
					24a7f3a320 | ||
| 
						 | 
					d3650f1145 | ||
| 
						 | 
					4801db9050 | ||
| 
						 | 
					6445281658 | ||
| 
						 | 
					87719f5938 | ||
| 
						 | 
					5ba1ec433b | ||
| 
						 | 
					7c5b382458 | ||
| 
						 | 
					8960426128 | ||
| 
						 | 
					e1d47d5a92 | ||
| 
						 | 
					9407c272aa | ||
| 
						 | 
					bccbd7b400 | ||
| 
						 | 
					a24dc010ec | ||
| 
						 | 
					c5e5d50fb8 | ||
| 
						 | 
					be1c520320 | ||
| 
						 | 
					3d18d0f893 | ||
| 
						 | 
					c3351a38a6 | ||
| 
						 | 
					0b86340a8d | ||
| 
						 | 
					829371b032 | ||
| 
						 | 
					b342207bc7 | ||
| 
						 | 
					58f4fdced3 | ||
| 
						 | 
					253844cbcf | ||
| 
						 | 
					e2e2b9ffb4 | ||
| 
						 | 
					2568042f5f | ||
| 
						 | 
					722edf4b9a | ||
| 
						 | 
					e406d364e4 | ||
| 
						 | 
					31c37f41b2 | ||
| 
						 | 
					6cd879bbc5 | ||
| 
						 | 
					b7974050fe | ||
| 
						 | 
					e9c9d0816e | ||
| 
						 | 
					a437692e1a | ||
| 
						 | 
					2a14d2f3c8 | ||
| 
						 | 
					559e2d1889 | ||
| 
						 | 
					a1aa919f80 | ||
| 
						 | 
					cbccb27a5a | ||
| 
						 | 
					bbad36d576 | ||
| 
						 | 
					df6c9d55b5 | ||
| 
						 | 
					36a1d9c364 | ||
| 
						 | 
					1f3681d5ac | ||
| 
						 | 
					0159f8e53f | ||
| 
						 | 
					0432be64fc | ||
| 
						 | 
					481e062961 | ||
| 
						 | 
					72b8abdeb6 | ||
| 
						 | 
					7ec3ee41d1 | ||
| 
						 | 
					b765fa4769 | ||
| 
						 | 
					06685b162e | ||
| 
						 | 
					7bcad7c424 | ||
| 
						 | 
					6e054b3cc6 | ||
| 
						 | 
					e643f6b0f8 | ||
| 
						 | 
					e44c0f85c2 | ||
| 
						 | 
					ea66e968eb | ||
| 
						 | 
					f7e73d804e | ||
| 
						 | 
					2ff4df56a1 | ||
| 
						 | 
					346c45fe0c | ||
| 
						 | 
					f98b42a36e | ||
| 
						 | 
					f77a0a266c | ||
| 
						 | 
					0be672788f | ||
| 
						 | 
					e2e3d11d42 | ||
| 
						 | 
					bb127bb567 | ||
| 
						 | 
					8eb4f89db8 | ||
| 
						 | 
					4a87ea3e70 | ||
| 
						 | 
					61115fce99 | ||
| 
						 | 
					4f31120394 | ||
| 
						 | 
					b085cd65ce | ||
| 
						 | 
					f76ec0721a | ||
| 
						 | 
					ef3944fbbf | ||
| 
						 | 
					b158fbc0d8 | ||
| 
						 | 
					68f2d66e97 | ||
| 
						 | 
					0cdd1735bd | ||
| 
						 | 
					70f5ead20b | ||
| 
						 | 
					574e5616f8 | ||
| 
						 | 
					94fa810590 | ||
| 
						 | 
					37f69da701 | ||
| 
						 | 
					f2f8448ade | ||
| 
						 | 
					b357d3fff2 | ||
| 
						 | 
					b417194905 | ||
| 
						 | 
					d8741da20a | ||
| 
						 | 
					c469be9a62 | ||
| 
						 | 
					ff4eb339ef | ||
| 
						 | 
					78489383c0 | ||
| 
						 | 
					f67c1b415f | ||
| 
						 | 
					570c16d921 | ||
| 
						 | 
					c8bc60568a | ||
| 
						 | 
					73de3ba856 | ||
| 
						 | 
					6a69be8537 | ||
| 
						 | 
					3ee381d505 | ||
| 
						 | 
					7ce744e2e4 | ||
| 
						 | 
					b81d21c991 | ||
| 
						 | 
					e9fe0992c6 | ||
| 
						 | 
					d5ef9018fa | ||
| 
						 | 
					c2737a7c51 | ||
| 
						 | 
					4b6d8733c6 | ||
| 
						 | 
					f22a3e0955 | ||
| 
						 | 
					4b8144a2f7 | ||
| 
						 | 
					abac52e23c | ||
| 
						 | 
					9aa089313e | ||
| 
						 | 
					b18f2c481a | ||
| 
						 | 
					ec83b9f77b | ||
| 
						 | 
					fad5495e02 | ||
| 
						 | 
					70dec1171e | ||
| 
						 | 
					0673a6fce3 | ||
| 
						 | 
					8f525b1407 | ||
| 
						 | 
					e83a991a4b | ||
| 
						 | 
					c000432a52 | ||
| 
						 | 
					c06d2d6927 | ||
| 
						 | 
					aa29653a8f | ||
| 
						 | 
					3f7e4ad486 | ||
| 
						 | 
					0eab546b2f | ||
| 
						 | 
					8830d216d1 | ||
| 
						 | 
					9fa1e3d449 | ||
| 
						 | 
					fe5dce7159 | ||
| 
						 | 
					952fa31548 | ||
| 
						 | 
					cc058ccc61 | ||
| 
						 | 
					bc26ed9701 | ||
| 
						 | 
					26135fc1a0 | ||
| 
						 | 
					0bd14672ff | ||
| 
						 | 
					7c7150cde8 | ||
| 
						 | 
					b08282b0c1 | ||
| 
						 | 
					3bf6e1befc | ||
| 
						 | 
					52692371ac | ||
| 
						 | 
					159ac1d8bb | ||
| 
						 | 
					7f1ffdbc79 | ||
| 
						 | 
					5fe64931dc | ||
| 
						 | 
					ee67855b48 | ||
| 
						 | 
					594a97f43f | ||
| 
						 | 
					7fb435a8b4 | ||
| 
						 | 
					01e74d6116 | ||
| 
						 | 
					b321d75b39 | ||
| 
						 | 
					1e47a45723 | ||
| 
						 | 
					88c609e5ef | ||
| 
						 | 
					775d1a424e | 
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -363,4 +363,6 @@ MigrationBackup/
 | 
			
		||||
FodyWeavers.xsd
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/framework/*pro*
 | 
			
		||||
/framework/*Pro*
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,7 +0,0 @@
 | 
			
		||||
[*.cs]
 | 
			
		||||
 | 
			
		||||
# CA1848: 使用 LoggerMessage 委托
 | 
			
		||||
dotnet_diagnostic.CA1848.severity = none
 | 
			
		||||
 | 
			
		||||
# CA2254: 模板应为静态表达式
 | 
			
		||||
dotnet_diagnostic.CA2254.severity = suggestion
 | 
			
		||||
							
								
								
									
										63
									
								
								framework/.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										63
									
								
								framework/.gitattributes
									
									
									
									
										vendored
									
									
								
							@@ -1,63 +0,0 @@
 | 
			
		||||
###############################################################################
 | 
			
		||||
# Set default behavior to automatically normalize line endings.
 | 
			
		||||
###############################################################################
 | 
			
		||||
* text=auto
 | 
			
		||||
 | 
			
		||||
###############################################################################
 | 
			
		||||
# Set default behavior for command prompt diff.
 | 
			
		||||
#
 | 
			
		||||
# This is need for earlier builds of msysgit that does not have it on by
 | 
			
		||||
# default for csharp files.
 | 
			
		||||
# Note: This is only used by command line
 | 
			
		||||
###############################################################################
 | 
			
		||||
#*.cs     diff=csharp
 | 
			
		||||
 | 
			
		||||
###############################################################################
 | 
			
		||||
# Set the merge driver for project and solution files
 | 
			
		||||
#
 | 
			
		||||
# Merging from the command prompt will add diff markers to the files if there
 | 
			
		||||
# are conflicts (Merging from VS is not affected by the settings below, in VS
 | 
			
		||||
# the diff markers are never inserted). Diff markers may cause the following 
 | 
			
		||||
# file extensions to fail to load in VS. An alternative would be to treat
 | 
			
		||||
# these files as binary and thus will always conflict and require user
 | 
			
		||||
# intervention with every merge. To do so, just uncomment the entries below
 | 
			
		||||
###############################################################################
 | 
			
		||||
#*.sln       merge=binary
 | 
			
		||||
#*.csproj    merge=binary
 | 
			
		||||
#*.vbproj    merge=binary
 | 
			
		||||
#*.vcxproj   merge=binary
 | 
			
		||||
#*.vcproj    merge=binary
 | 
			
		||||
#*.dbproj    merge=binary
 | 
			
		||||
#*.fsproj    merge=binary
 | 
			
		||||
#*.lsproj    merge=binary
 | 
			
		||||
#*.wixproj   merge=binary
 | 
			
		||||
#*.modelproj merge=binary
 | 
			
		||||
#*.sqlproj   merge=binary
 | 
			
		||||
#*.wwaproj   merge=binary
 | 
			
		||||
 | 
			
		||||
###############################################################################
 | 
			
		||||
# behavior for image files
 | 
			
		||||
#
 | 
			
		||||
# image files are treated as binary by default.
 | 
			
		||||
###############################################################################
 | 
			
		||||
#*.jpg   binary
 | 
			
		||||
#*.png   binary
 | 
			
		||||
#*.gif   binary
 | 
			
		||||
 | 
			
		||||
###############################################################################
 | 
			
		||||
# diff behavior for common document formats
 | 
			
		||||
# 
 | 
			
		||||
# Convert binary document formats to text before diffing them. This feature
 | 
			
		||||
# is only available from the command line. Turn it on by uncommenting the 
 | 
			
		||||
# entries below.
 | 
			
		||||
###############################################################################
 | 
			
		||||
#*.doc   diff=astextplain
 | 
			
		||||
#*.DOC   diff=astextplain
 | 
			
		||||
#*.docx  diff=astextplain
 | 
			
		||||
#*.DOCX  diff=astextplain
 | 
			
		||||
#*.dot   diff=astextplain
 | 
			
		||||
#*.DOT   diff=astextplain
 | 
			
		||||
#*.pdf   diff=astextplain
 | 
			
		||||
#*.PDF   diff=astextplain
 | 
			
		||||
#*.rtf   diff=astextplain
 | 
			
		||||
#*.RTF   diff=astextplain
 | 
			
		||||
							
								
								
									
										364
									
								
								framework/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										364
									
								
								framework/.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -1,364 +0,0 @@
 | 
			
		||||
## Ignore Visual Studio temporary files, build results, and
 | 
			
		||||
## files generated by popular Visual Studio add-ons.
 | 
			
		||||
##
 | 
			
		||||
## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore
 | 
			
		||||
 | 
			
		||||
# User-specific files
 | 
			
		||||
*.rsuser
 | 
			
		||||
*.suo
 | 
			
		||||
*.user
 | 
			
		||||
*.userosscache
 | 
			
		||||
*.sln.docstates
 | 
			
		||||
 | 
			
		||||
# User-specific files (MonoDevelop/Xamarin Studio)
 | 
			
		||||
*.userprefs
 | 
			
		||||
 | 
			
		||||
# Mono auto generated files
 | 
			
		||||
mono_crash.*
 | 
			
		||||
 | 
			
		||||
# Build results
 | 
			
		||||
[Dd]ebug/
 | 
			
		||||
[Dd]ebugPublic/
 | 
			
		||||
[Rr]elease/
 | 
			
		||||
[Rr]eleases/
 | 
			
		||||
x64/
 | 
			
		||||
x86/
 | 
			
		||||
[Ww][Ii][Nn]32/
 | 
			
		||||
[Aa][Rr][Mm]/
 | 
			
		||||
[Aa][Rr][Mm]64/
 | 
			
		||||
bld/
 | 
			
		||||
[Bb]in/
 | 
			
		||||
[Oo]bj/
 | 
			
		||||
[Oo]ut/
 | 
			
		||||
[Ll]og/
 | 
			
		||||
[Ll]ogs/
 | 
			
		||||
 | 
			
		||||
# Visual Studio 2015/2017 cache/options directory
 | 
			
		||||
.vs/
 | 
			
		||||
# Uncomment if you have tasks that create the project's static files in wwwroot
 | 
			
		||||
#wwwroot/
 | 
			
		||||
 | 
			
		||||
# Visual Studio 2017 auto generated files
 | 
			
		||||
Generated\ Files/
 | 
			
		||||
 | 
			
		||||
# MSTest test Results
 | 
			
		||||
[Tt]est[Rr]esult*/
 | 
			
		||||
[Bb]uild[Ll]og.*
 | 
			
		||||
 | 
			
		||||
# NUnit
 | 
			
		||||
*.VisualState.xml
 | 
			
		||||
TestResult.xml
 | 
			
		||||
nunit-*.xml
 | 
			
		||||
 | 
			
		||||
# Build Results of an ATL Project
 | 
			
		||||
[Dd]ebugPS/
 | 
			
		||||
[Rr]eleasePS/
 | 
			
		||||
dlldata.c
 | 
			
		||||
 | 
			
		||||
# Benchmark Results
 | 
			
		||||
BenchmarkDotNet.Artifacts/
 | 
			
		||||
 | 
			
		||||
# .NET Core
 | 
			
		||||
project.lock.json
 | 
			
		||||
project.fragment.lock.json
 | 
			
		||||
artifacts/
 | 
			
		||||
 | 
			
		||||
# ASP.NET Scaffolding
 | 
			
		||||
ScaffoldingReadMe.txt
 | 
			
		||||
 | 
			
		||||
# StyleCop
 | 
			
		||||
StyleCopReport.xml
 | 
			
		||||
 | 
			
		||||
# Files built by Visual Studio
 | 
			
		||||
*_i.c
 | 
			
		||||
*_p.c
 | 
			
		||||
*_h.h
 | 
			
		||||
*.ilk
 | 
			
		||||
*.meta
 | 
			
		||||
*.obj
 | 
			
		||||
*.iobj
 | 
			
		||||
*.pch
 | 
			
		||||
*.pdb
 | 
			
		||||
*.ipdb
 | 
			
		||||
*.pgc
 | 
			
		||||
*.pgd
 | 
			
		||||
*.rsp
 | 
			
		||||
*.sbr
 | 
			
		||||
*.tlb
 | 
			
		||||
*.tli
 | 
			
		||||
*.tlh
 | 
			
		||||
*.tmp
 | 
			
		||||
*.tmp_proj
 | 
			
		||||
*_wpftmp.csproj
 | 
			
		||||
*.log
 | 
			
		||||
*.vspscc
 | 
			
		||||
*.vssscc
 | 
			
		||||
.builds
 | 
			
		||||
*.pidb
 | 
			
		||||
*.svclog
 | 
			
		||||
*.scc
 | 
			
		||||
 | 
			
		||||
# Chutzpah Test files
 | 
			
		||||
_Chutzpah*
 | 
			
		||||
 | 
			
		||||
# Visual C++ cache files
 | 
			
		||||
ipch/
 | 
			
		||||
*.aps
 | 
			
		||||
*.ncb
 | 
			
		||||
*.opendb
 | 
			
		||||
*.opensdf
 | 
			
		||||
*.sdf
 | 
			
		||||
*.cachefile
 | 
			
		||||
*.VC.db
 | 
			
		||||
*.VC.VC.opendb
 | 
			
		||||
 | 
			
		||||
# Visual Studio profiler
 | 
			
		||||
*.psess
 | 
			
		||||
*.vsp
 | 
			
		||||
*.vspx
 | 
			
		||||
*.sap
 | 
			
		||||
 | 
			
		||||
# Visual Studio Trace Files
 | 
			
		||||
*.e2e
 | 
			
		||||
 | 
			
		||||
# TFS 2012 Local Workspace
 | 
			
		||||
$tf/
 | 
			
		||||
 | 
			
		||||
# Guidance Automation Toolkit
 | 
			
		||||
*.gpState
 | 
			
		||||
 | 
			
		||||
# ReSharper is a .NET coding add-in
 | 
			
		||||
_ReSharper*/
 | 
			
		||||
*.[Rr]e[Ss]harper
 | 
			
		||||
*.DotSettings.user
 | 
			
		||||
 | 
			
		||||
# TeamCity is a build add-in
 | 
			
		||||
_TeamCity*
 | 
			
		||||
 | 
			
		||||
# DotCover is a Code Coverage Tool
 | 
			
		||||
*.dotCover
 | 
			
		||||
 | 
			
		||||
# AxoCover is a Code Coverage Tool
 | 
			
		||||
.axoCover/*
 | 
			
		||||
!.axoCover/settings.json
 | 
			
		||||
 | 
			
		||||
# Coverlet is a free, cross platform Code Coverage Tool
 | 
			
		||||
coverage*.json
 | 
			
		||||
coverage*.xml
 | 
			
		||||
coverage*.info
 | 
			
		||||
 | 
			
		||||
# Visual Studio code coverage results
 | 
			
		||||
*.coverage
 | 
			
		||||
*.coveragexml
 | 
			
		||||
 | 
			
		||||
# NCrunch
 | 
			
		||||
_NCrunch_*
 | 
			
		||||
.*crunch*.local.xml
 | 
			
		||||
nCrunchTemp_*
 | 
			
		||||
 | 
			
		||||
# MightyMoose
 | 
			
		||||
*.mm.*
 | 
			
		||||
AutoTest.Net/
 | 
			
		||||
 | 
			
		||||
# Web workbench (sass)
 | 
			
		||||
.sass-cache/
 | 
			
		||||
 | 
			
		||||
# Installshield output folder
 | 
			
		||||
[Ee]xpress/
 | 
			
		||||
 | 
			
		||||
# DocProject is a documentation generator add-in
 | 
			
		||||
DocProject/buildhelp/
 | 
			
		||||
DocProject/Help/*.HxT
 | 
			
		||||
DocProject/Help/*.HxC
 | 
			
		||||
DocProject/Help/*.hhc
 | 
			
		||||
DocProject/Help/*.hhk
 | 
			
		||||
DocProject/Help/*.hhp
 | 
			
		||||
DocProject/Help/Html2
 | 
			
		||||
DocProject/Help/html
 | 
			
		||||
 | 
			
		||||
# Click-Once directory
 | 
			
		||||
publish/
 | 
			
		||||
 | 
			
		||||
# Publish Web Output
 | 
			
		||||
*.[Pp]ublish.xml
 | 
			
		||||
*.azurePubxml
 | 
			
		||||
# Note: Comment the next line if you want to checkin your web deploy settings,
 | 
			
		||||
# but database connection strings (with potential passwords) will be unencrypted
 | 
			
		||||
*.pubxml
 | 
			
		||||
*.publishproj
 | 
			
		||||
 | 
			
		||||
# Microsoft Azure Web App publish settings. Comment the next line if you want to
 | 
			
		||||
# checkin your Azure Web App publish settings, but sensitive information contained
 | 
			
		||||
# in these scripts will be unencrypted
 | 
			
		||||
PublishScripts/
 | 
			
		||||
 | 
			
		||||
# NuGet Packages
 | 
			
		||||
*.nupkg
 | 
			
		||||
# NuGet Symbol Packages
 | 
			
		||||
*.snupkg
 | 
			
		||||
# The packages folder can be ignored because of Package Restore
 | 
			
		||||
**/[Pp]ackages/*
 | 
			
		||||
# except build/, which is used as an MSBuild target.
 | 
			
		||||
!**/[Pp]ackages/build/
 | 
			
		||||
# Uncomment if necessary however generally it will be regenerated when needed
 | 
			
		||||
#!**/[Pp]ackages/repositories.config
 | 
			
		||||
# NuGet v3's project.json files produces more ignorable files
 | 
			
		||||
*.nuget.props
 | 
			
		||||
*.nuget.targets
 | 
			
		||||
 | 
			
		||||
# Microsoft Azure Build Output
 | 
			
		||||
csx/
 | 
			
		||||
*.build.csdef
 | 
			
		||||
 | 
			
		||||
# Microsoft Azure Emulator
 | 
			
		||||
ecf/
 | 
			
		||||
rcf/
 | 
			
		||||
 | 
			
		||||
# Windows Store app package directories and files
 | 
			
		||||
AppPackages/
 | 
			
		||||
BundleArtifacts/
 | 
			
		||||
Package.StoreAssociation.xml
 | 
			
		||||
_pkginfo.txt
 | 
			
		||||
*.appx
 | 
			
		||||
*.appxbundle
 | 
			
		||||
*.appxupload
 | 
			
		||||
 | 
			
		||||
# Visual Studio cache files
 | 
			
		||||
# files ending in .cache can be ignored
 | 
			
		||||
*.[Cc]ache
 | 
			
		||||
# but keep track of directories ending in .cache
 | 
			
		||||
!?*.[Cc]ache/
 | 
			
		||||
 | 
			
		||||
# Others
 | 
			
		||||
ClientBin/
 | 
			
		||||
~$*
 | 
			
		||||
*~
 | 
			
		||||
*.dbmdl
 | 
			
		||||
*.dbproj.schemaview
 | 
			
		||||
*.jfm
 | 
			
		||||
*.pfx
 | 
			
		||||
*.publishsettings
 | 
			
		||||
orleans.codegen.cs
 | 
			
		||||
 | 
			
		||||
# Including strong name files can present a security risk
 | 
			
		||||
# (https://github.com/github/gitignore/pull/2483#issue-259490424)
 | 
			
		||||
#*.snk
 | 
			
		||||
 | 
			
		||||
# Since there are multiple workflows, uncomment next line to ignore bower_components
 | 
			
		||||
# (https://github.com/github/gitignore/pull/1529#issuecomment-104372622)
 | 
			
		||||
#bower_components/
 | 
			
		||||
 | 
			
		||||
# RIA/Silverlight projects
 | 
			
		||||
Generated_Code/
 | 
			
		||||
 | 
			
		||||
# Backup & report files from converting an old project file
 | 
			
		||||
# to a newer Visual Studio version. Backup files are not needed,
 | 
			
		||||
# because we have git ;-)
 | 
			
		||||
_UpgradeReport_Files/
 | 
			
		||||
Backup*/
 | 
			
		||||
UpgradeLog*.XML
 | 
			
		||||
UpgradeLog*.htm
 | 
			
		||||
ServiceFabricBackup/
 | 
			
		||||
*.rptproj.bak
 | 
			
		||||
 | 
			
		||||
# SQL Server files
 | 
			
		||||
*.mdf
 | 
			
		||||
*.ldf
 | 
			
		||||
*.ndf
 | 
			
		||||
 | 
			
		||||
# Business Intelligence projects
 | 
			
		||||
*.rdl.data
 | 
			
		||||
*.bim.layout
 | 
			
		||||
*.bim_*.settings
 | 
			
		||||
*.rptproj.rsuser
 | 
			
		||||
*- [Bb]ackup.rdl
 | 
			
		||||
*- [Bb]ackup ([0-9]).rdl
 | 
			
		||||
*- [Bb]ackup ([0-9][0-9]).rdl
 | 
			
		||||
 | 
			
		||||
# Microsoft Fakes
 | 
			
		||||
FakesAssemblies/
 | 
			
		||||
 | 
			
		||||
# GhostDoc plugin setting file
 | 
			
		||||
*.GhostDoc.xml
 | 
			
		||||
 | 
			
		||||
# Node.js Tools for Visual Studio
 | 
			
		||||
.ntvs_analysis.dat
 | 
			
		||||
node_modules/
 | 
			
		||||
 | 
			
		||||
# Visual Studio 6 build log
 | 
			
		||||
*.plg
 | 
			
		||||
 | 
			
		||||
# Visual Studio 6 workspace options file
 | 
			
		||||
*.opt
 | 
			
		||||
 | 
			
		||||
# Visual Studio 6 auto-generated workspace file (contains which files were open etc.)
 | 
			
		||||
*.vbw
 | 
			
		||||
 | 
			
		||||
# Visual Studio LightSwitch build output
 | 
			
		||||
**/*.HTMLClient/GeneratedArtifacts
 | 
			
		||||
**/*.DesktopClient/GeneratedArtifacts
 | 
			
		||||
**/*.DesktopClient/ModelManifest.xml
 | 
			
		||||
**/*.Server/GeneratedArtifacts
 | 
			
		||||
**/*.Server/ModelManifest.xml
 | 
			
		||||
_Pvt_Extensions
 | 
			
		||||
 | 
			
		||||
# Paket dependency manager
 | 
			
		||||
.paket/paket.exe
 | 
			
		||||
paket-files/
 | 
			
		||||
 | 
			
		||||
# FAKE - F# Make
 | 
			
		||||
.fake/
 | 
			
		||||
 | 
			
		||||
# CodeRush personal settings
 | 
			
		||||
.cr/personal
 | 
			
		||||
 | 
			
		||||
# Python Tools for Visual Studio (PTVS)
 | 
			
		||||
__pycache__/
 | 
			
		||||
*.pyc
 | 
			
		||||
 | 
			
		||||
# Cake - Uncomment if you are using it
 | 
			
		||||
# tools/**
 | 
			
		||||
# !tools/packages.config
 | 
			
		||||
 | 
			
		||||
# Tabs Studio
 | 
			
		||||
*.tss
 | 
			
		||||
 | 
			
		||||
# Telerik's JustMock configuration file
 | 
			
		||||
*.jmconfig
 | 
			
		||||
 | 
			
		||||
# BizTalk build output
 | 
			
		||||
*.btp.cs
 | 
			
		||||
*.btm.cs
 | 
			
		||||
*.odx.cs
 | 
			
		||||
*.xsd.cs
 | 
			
		||||
 | 
			
		||||
# OpenCover UI analysis results
 | 
			
		||||
OpenCover/
 | 
			
		||||
 | 
			
		||||
# Azure Stream Analytics local run output
 | 
			
		||||
ASALocalRun/
 | 
			
		||||
 | 
			
		||||
# MSBuild Binary and Structured Log
 | 
			
		||||
*.binlog
 | 
			
		||||
 | 
			
		||||
# NVidia Nsight GPU debugger configuration file
 | 
			
		||||
*.nvuser
 | 
			
		||||
 | 
			
		||||
# MFractors (Xamarin productivity tool) working folder
 | 
			
		||||
.mfractor/
 | 
			
		||||
 | 
			
		||||
# Local History for Visual Studio
 | 
			
		||||
.localhistory/
 | 
			
		||||
 | 
			
		||||
# BeatPulse healthcheck temp database
 | 
			
		||||
healthchecksdb
 | 
			
		||||
 | 
			
		||||
# Backup folder for Package Reference Convert tool in Visual Studio 2017
 | 
			
		||||
MigrationBackup/
 | 
			
		||||
 | 
			
		||||
# Ionide (cross platform F# VS Code tools) working folder
 | 
			
		||||
.ionide/
 | 
			
		||||
 | 
			
		||||
# Fody - auto-generated XML schema
 | 
			
		||||
FodyWeavers.xsd
 | 
			
		||||
 | 
			
		||||
@@ -1,15 +0,0 @@
 | 
			
		||||
<Project>
 | 
			
		||||
	<PropertyGroup>
 | 
			
		||||
		<Version>3.0.0.12</Version>
 | 
			
		||||
		<LangVersion>latest</LangVersion>
 | 
			
		||||
		<TargetFrameworks>net6.0;net7.0</TargetFrameworks>
 | 
			
		||||
		<Authors>Diego</Authors>
 | 
			
		||||
		<Product>ThingsGateway</Product>
 | 
			
		||||
		<Copyright>© 2023-present Diego</Copyright>
 | 
			
		||||
		<RepositoryUrl>https://gitee.com/diego2098/ThingsGateway</RepositoryUrl>
 | 
			
		||||
		<SatelliteResourceLanguages>zh-Hans</SatelliteResourceLanguages>
 | 
			
		||||
		<GenerateDocumentationFile>False</GenerateDocumentationFile>
 | 
			
		||||
	</PropertyGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
</Project>
 | 
			
		||||
@@ -1,152 +0,0 @@
 | 
			
		||||
<Project Sdk="Microsoft.NET.Sdk.Razor">
 | 
			
		||||
 | 
			
		||||
	<PropertyGroup Condition="'$(SolutionName)'=='ThingsGateway - Pro'">
 | 
			
		||||
		<DefineConstants>Pro</DefineConstants>
 | 
			
		||||
	</PropertyGroup>
 | 
			
		||||
	<ItemGroup Condition="'$(SolutionName)'=='ThingsGateway - Pro'">
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
		<Compile Include="..\..\PluginPro\ThingsGateway.Plugin.Melsec\Page\QnA3E_BinaryDebugPage.razor.cs" Link="Pages\Melsec\QnA3E_BinaryDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\PluginPro\ThingsGateway.Plugin.Melsec\Page\QnA3E_BinaryDebugPage.razor" Link="Pages\Melsec\QnA3E_BinaryDebugPage.razor" />
 | 
			
		||||
		<ProjectReference Include="..\..\FoundationPro\ThingsGateway.Foundation.Adapter.Melsec\ThingsGateway.Foundation.Adapter.Melsec.csproj" />
 | 
			
		||||
 | 
			
		||||
		
 | 
			
		||||
		
 | 
			
		||||
		<Compile Include="..\..\PluginPro\ThingsGateway.Plugin.AllenBradleyCip\Page\AllenBradleyCipTcpDebugPage.razor.cs" Link="Pages\ABCIP\AllenBradleyCipTcpDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\PluginPro\ThingsGateway.Plugin.AllenBradleyCip\Page\AllenBradleyCipTcpDebugPage.razor" Link="Pages\ABCIP\AllenBradleyCipTcpDebugPage.razor" />
 | 
			
		||||
		<ProjectReference Include="..\..\FoundationPro\ThingsGateway.Foundation.Adapter.AllenBradleyCip\ThingsGateway.Foundation.Adapter.AllenBradleyCip.csproj" />
 | 
			
		||||
 | 
			
		||||
		<Compile Include="..\..\PluginPro\ThingsGateway.Plugin.Omron\Page\OmronFinsTcpDebugPage.razor.cs" Link="Pages\OmronFins\OmronFinsTcpDebugPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\PluginPro\ThingsGateway.Plugin.Omron\Page\OmronFinsUdpDebugPage.razor.cs" Link="Pages\OmronFins\OmronFinsUdpDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\PluginPro\ThingsGateway.Plugin.Omron\Page\OmronFinsTcpDebugPage.razor" Link="Pages\OmronFins\OmronFinsTcpDebugPage.razor" />
 | 
			
		||||
		<Content Include="..\..\PluginPro\ThingsGateway.Plugin.Omron\Page\OmronFinsUdpDebugPage.razor" Link="Pages\OmronFins\OmronFinsUdpDebugPage.razor" />
 | 
			
		||||
		<ProjectReference Include="..\..\FoundationPro\ThingsGateway.Foundation.Adapter.Omron\ThingsGateway.Foundation.Adapter.Omron.csproj" />
 | 
			
		||||
 | 
			
		||||
		<Compile Include="..\..\PluginPro\ThingsGateway.Plugin.Secs\Page\SecsHsmsTcpDebugPage.razor.cs" Link="Pages\Secs\SecsHsmsTcpDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\PluginPro\ThingsGateway.Plugin.Secs\Page\SecsHsmsTcpDebugPage.razor" Link="Pages\Secs\SecsHsmsTcpDebugPage.razor" />
 | 
			
		||||
		<ProjectReference Include="..\..\FoundationPro\ThingsGateway.Foundation.Adapter.Secs\ThingsGateway.Foundation.Adapter.Secs.csproj" />
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
		<Compile Include="..\..\PluginPro\ThingsGateway.Plugin.TS550\Page\TS550DebugPage.razor.cs" Link="Pages\TS550\TS550DebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\PluginPro\ThingsGateway.Plugin.TS550\Page\TS550DebugPage.razor" Link="Pages\TS550\TS550DebugPage.razor" />
 | 
			
		||||
		<ProjectReference Include="..\..\FoundationPro\ThingsGateway.Foundation.Adapter.TS550\ThingsGateway.Foundation.Adapter.TS550.csproj" />
 | 
			
		||||
 | 
			
		||||
		<Compile Include="..\..\PluginPro\ThingsGateway.Plugin.Vigor\Page\VigorSerialDebugPage.razor.cs" Link="Pages\Vigor\VigorSerialDebugPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\PluginPro\ThingsGateway.Plugin.Vigor\Page\VigorSerialOverTcpDebugPage.razor.cs" Link="Pages\Vigor\VigorSerialOverTcpDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\PluginPro\ThingsGateway.Plugin.Vigor\Page\VigorSerialDebugPage.razor" Link="Pages\Vigor\VigorSerialDebugPage.razor" />
 | 
			
		||||
		<Content Include="..\..\PluginPro\ThingsGateway.Plugin.Vigor\Page\VigorSerialOverTcpDebugPage.razor" Link="Pages\Vigor\VigorSerialOverTcpDebugPage.razor" />
 | 
			
		||||
		<ProjectReference Include="..\..\FoundationPro\ThingsGateway.Foundation.Adapter.Vigor\ThingsGateway.Foundation.Adapter.Vigor.csproj" />
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
		<Compile Include="..\..\PluginProAF2021\ThingsGateway.Plugin.HZW_QTJC_01\Page\HZW_QTJC_01SerialDebugPage.razor.cs" Link="Pages\HZW_QTJC_01\HZW_QTJC_01SerialDebugPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\PluginProAF2021\ThingsGateway.Plugin.HZW_QTJC_01\Page\HZW_QTJC_01SerialOverTcpDebugPage.razor.cs" Link="Pages\HZW_QTJC_01\HZW_QTJC_01SerialOverTcpDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\PluginProAF2021\ThingsGateway.Plugin.HZW_QTJC_01\Page\HZW_QTJC_01SerialDebugPage.razor" Link="Pages\HZW_QTJC_01\HZW_QTJC_01SerialDebugPage.razor" />
 | 
			
		||||
		<Content Include="..\..\PluginProAF2021\ThingsGateway.Plugin.HZW_QTJC_01\Page\HZW_QTJC_01SerialOverTcpDebugPage.razor" Link="Pages\HZW_QTJC_01\HZW_QTJC_01SerialOverTcpDebugPage.razor" />
 | 
			
		||||
		<ProjectReference Include="..\..\PluginProAF2021\ThingsGateway.Foundation.Adapter.HZW_QTJC_01\ThingsGateway.Foundation.Adapter.HZW_QTJC_01.csproj" />
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
		
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.DLT645\Page\DLT645_2007DebugPage.razor.cs" Link="Pages\DLT645\DLT645_2007DebugPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Mqtt\MqttRpcNameVaueWithId.cs" Link="Pages\Mqtt\MqttRpcNameVaueWithId.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Mqtt\Page\MqttClientDebugPage.razor.cs" Link="Pages\Mqtt\MqttClientDebugPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Mqtt\Page\MqttClientPage.razor.cs" Link="Pages\Mqtt\MqttClientPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Mqtt\PrivateLogger.cs" Link="Pages\Mqtt\PrivateLogger.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Mqtt\RpcClass\MqttRpcClient.cs" Link="Pages\Mqtt\MqttRpcClient.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Mqtt\RpcClass\MqttRpcClientExtensions.cs" Link="Pages\Mqtt\MqttRpcClientExtensions.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Mqtt\RpcClass\MqttRpcTopicPair.cs" Link="Pages\Mqtt\MqttRpcTopicPair.cs" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.DLT645\Page\DLT645_2007DebugPage.razor" Link="Pages\DLT645\DLT645_2007DebugPage.razor" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.DLT645\Page\DLT645_2007OverTcpDebugPage.razor.cs" Link="Pages\DLT645\DLT645_2007OverTcpDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.DLT645\Page\DLT645_2007OverTcpDebugPage.razor" Link="Pages\DLT645\DLT645_2007OverTcpDebugPage.razor" />
 | 
			
		||||
 | 
			
		||||
		
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusRtuDebugPage.razor.cs" Link="Pages\Modbus\ModbusRtuDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusRtuDebugPage.razor" Link="Pages\Modbus\ModbusRtuDebugPage.razor" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusRtuOverTcpDebugPage.razor.cs" Link="Pages\Modbus\ModbusRtuOverTcpDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusRtuOverTcpDebugPage.razor" Link="Pages\Modbus\ModbusRtuOverTcpDebugPage.razor" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusRtuOverUdpDebugPage.razor.cs" Link="Pages\Modbus\ModbusRtuOverUdpDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusRtuOverUdpDebugPage.razor" Link="Pages\Modbus\ModbusRtuOverUdpDebugPage.razor" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusSerialServerDebugPage.razor.cs" Link="Pages\Modbus\ModbusSerialServerDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusSerialServerDebugPage.razor" Link="Pages\Modbus\ModbusSerialServerDebugPage.razor" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusTcpDebugPage.razor.cs" Link="Pages\Modbus\ModbusTcpDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusTcpDebugPage.razor" Link="Pages\Modbus\ModbusTcpDebugPage.razor" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusTcpDtuDebugPage.razor.cs" Link="Pages\Modbus\ModbusTcpDtuDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusTcpDtuDebugPage.razor" Link="Pages\Modbus\ModbusTcpDtuDebugPage.razor" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusTcpServerDebugPage.razor.cs" Link="Pages\Modbus\ModbusTcpServerDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusTcpServerDebugPage.razor" Link="Pages\Modbus\ModbusTcpServerDebugPage.razor" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusUdpDebugPage.razor.cs" Link="Pages\Modbus\ModbusUdpDebugPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Modbus\Page\ModbusUdpDebugPage.razor" Link="Pages\Modbus\ModbusUdpDebugPage.razor" />
 | 
			
		||||
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.OPCDA\Page\OPCDAClientDebugPage.razor.cs" Link="Pages\OPCDA\OPCDAClientDebugPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.OPCDA\Page\OPCDAClientPage.razor.cs" Link="Pages\OPCDA\OPCDAClientPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.OPCDA\Page\OPCDAImportVariable.razor.cs" Link="Pages\OPCDA\OPCDAImportVariable.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.OPCDA\Page\OPCDAClientDebugPage.razor" Link="Pages\OPCDA\OPCDAClientDebugPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.OPCDA\Page\OPCDAClientPage.razor" Link="Pages\OPCDA\OPCDAClientPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.OPCDA\Page\OPCDAImportVariable.razor" Link="Pages\OPCDA\OPCDAImportVariable.razor" />
 | 
			
		||||
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.OPCUA\Page\OPCUAClientDebugPage.razor" Link="Pages\OPCUA\OPCUAClientDebugPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.OPCUA\Page\OPCUAClientPage.razor" Link="Pages\OPCUA\OPCUAClientPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.OPCUA\Page\OPCUAImportVariable.razor" Link="Pages\OPCUA\OPCUAImportVariable.razor" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.OPCUA\Page\OPCUAClientDebugPage.razor.cs" Link="Pages\OPCUA\OPCUAClientDebugPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.OPCUA\Page\OPCUAClientPage.razor.cs" Link="Pages\OPCUA\OPCUAClientPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.OPCUA\Page\OPCUAImportVariable.razor.cs" Link="Pages\OPCUA\OPCUAImportVariable.razor.cs" />
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Siemens\Page\S7_1200DebugPage.razor" Link="Pages\Siemens\S7_1200DebugPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Siemens\Page\S7_1500DebugPage.razor" Link="Pages\Siemens\S7_1500DebugPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Siemens\Page\S7_200DebugPage.razor" Link="Pages\Siemens\S7_200DebugPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Siemens\Page\S7_200SMARTDebugPage.razor" Link="Pages\Siemens\S7_200SMARTDebugPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Siemens\Page\S7_300DebugPage.razor" Link="Pages\Siemens\S7_300DebugPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Plugin\ThingsGateway.Plugin.Siemens\Page\S7_400DebugPage.razor" Link="Pages\Siemens\S7_400DebugPage.razor" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Siemens\Page\S7_1200DebugPage.razor.cs" Link="Pages\Siemens\S7_1200DebugPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Siemens\Page\S7_1500DebugPage.razor.cs" Link="Pages\Siemens\S7_1500DebugPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Siemens\Page\S7_200DebugPage.razor.cs" Link="Pages\Siemens\S7_200DebugPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Siemens\Page\S7_200SMARTDebugPage.razor.cs" Link="Pages\Siemens\S7_200SMARTDebugPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Siemens\Page\S7_300DebugPage.razor.cs" Link="Pages\Siemens\S7_300DebugPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Plugin\ThingsGateway.Plugin.Siemens\Page\S7_400DebugPage.razor.cs" Link="Pages\Siemens\S7_400DebugPage.razor.cs" />
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	<ItemGroup >
 | 
			
		||||
		<PackageReference Include="ThingsGateway.Foundation.Adapter.DLT645" Version="*" />
 | 
			
		||||
		<PackageReference Include="ThingsGateway.Foundation.Adapter.Modbus" Version="*" />
 | 
			
		||||
		<PackageReference Include="ThingsGateway.Foundation.Adapter.OPCDA" Version="*" />
 | 
			
		||||
		<PackageReference Include="ThingsGateway.Foundation.Adapter.OPCUA" Version="*" />
 | 
			
		||||
		<PackageReference Include="ThingsGateway.Foundation.Adapter.Siemens" Version="*" />
 | 
			
		||||
		<!--<ProjectReference Include="..\..\Foundation\ThingsGateway.Foundation.Adapter.DLT645\ThingsGateway.Foundation.Adapter.DLT645.csproj" />
 | 
			
		||||
		<ProjectReference Include="..\..\Foundation\ThingsGateway.Foundation.Adapter.Modbus\ThingsGateway.Foundation.Adapter.Modbus.csproj" />
 | 
			
		||||
		<ProjectReference Include="..\..\Foundation\ThingsGateway.Foundation.Adapter.OPCDA\ThingsGateway.Foundation.Adapter.OPCDA.csproj" />
 | 
			
		||||
		<ProjectReference Include="..\..\Foundation\ThingsGateway.Foundation.Adapter.OPCUA\ThingsGateway.Foundation.Adapter.OPCUA.csproj" />
 | 
			
		||||
		<ProjectReference Include="..\..\Foundation\ThingsGateway.Foundation.Adapter.Siemens\ThingsGateway.Foundation.Adapter.Siemens.csproj" />-->
 | 
			
		||||
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
	<ItemGroup >
 | 
			
		||||
		<ProjectReference Include="..\..\Web\ThingsGateway.Components\ThingsGateway.Components.csproj" />
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
		<Content Update="wwwroot\**">
 | 
			
		||||
			<CopyToOutputDirectory>Always</CopyToOutputDirectory>
 | 
			
		||||
		</Content>
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
	  <Content Include="..\..\Plugin\ThingsGateway.Plugin.Mqtt\Page\MqttClientDebugPage.razor" Link="Pages\Mqtt\MqttClientDebugPage.razor" />
 | 
			
		||||
	  <Content Include="..\..\Plugin\ThingsGateway.Plugin.Mqtt\Page\MqttClientPage.razor" Link="Pages\Mqtt\MqttClientPage.razor" />
 | 
			
		||||
		<PackageReference Include="MQTTnet" Version="4.3.1.873" />
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
</Project>
 | 
			
		||||
@@ -1,9 +1,10 @@
 | 
			
		||||
<Project>
 | 
			
		||||
	<PropertyGroup>
 | 
			
		||||
		<Version>3.0.0.12</Version>
 | 
			
		||||
		<LangVersion>latest</LangVersion>
 | 
			
		||||
		
 | 
			
		||||
		<ImplicitUsings>enable</ImplicitUsings>
 | 
			
		||||
		<TargetFrameworks>net6.0;net7.0</TargetFrameworks>
 | 
			
		||||
		<TargetFrameworks>net6.0;net8.0;</TargetFrameworks>
 | 
			
		||||
		<Version>4.0.0.8</Version>
 | 
			
		||||
		<LangVersion>latest</LangVersion>
 | 
			
		||||
		<Authors>Diego</Authors>
 | 
			
		||||
		<Product>ThingsGateway</Product>
 | 
			
		||||
		<Copyright>© 2023-present Diego</Copyright>
 | 
			
		||||
@@ -11,7 +12,7 @@
 | 
			
		||||
		<SignAssembly>True</SignAssembly>
 | 
			
		||||
		<DelaySign>False</DelaySign>
 | 
			
		||||
		<SatelliteResourceLanguages>zh-Hans</SatelliteResourceLanguages>
 | 
			
		||||
		<GenerateDocumentationFile>True</GenerateDocumentationFile>
 | 
			
		||||
		<GenerateDocumentationFile>False</GenerateDocumentationFile>
 | 
			
		||||
	</PropertyGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -1,480 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.ComponentModel;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Extension.Generic;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Adapter.DLT645;
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// DLT645_2007
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class DLT645_2007 : ReadWriteDevicesSerialSessionBase
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// DLT645_2007
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="serialSession"></param>
 | 
			
		||||
    public DLT645_2007(SerialSession serialSession) : base(serialSession)
 | 
			
		||||
    {
 | 
			
		||||
        ThingsGatewayBitConverter = new DLT645_2007BitConverter(EndianType.Big);
 | 
			
		||||
        RegisterByteLength = 2;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 增加FE FE FE FE的报文头部
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("前导符报文头")]
 | 
			
		||||
    public bool EnableFEHead { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 写入需操作员代码
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("操作员代码")]
 | 
			
		||||
    public string OperCode { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 写入密码
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("写入密码")]
 | 
			
		||||
    public string Password { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 通讯地址BCD码,一般应该是12个字符
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("通讯地址")]
 | 
			
		||||
    public string Station { get; set; }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override string GetAddressDescription()
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
        var str = """
 | 
			
		||||
            查看附带文档或者相关资料,下面列举一下常见的数据标识地址 
 | 
			
		||||
            
 | 
			
		||||
            地址                       说明                    
 | 
			
		||||
            -----------------------------------------
 | 
			
		||||
            02010100    A相电压
 | 
			
		||||
            02020100    A相电流
 | 
			
		||||
            02030000    瞬时总有功功率
 | 
			
		||||
            00000000    (当前)组合有功总电能
 | 
			
		||||
            00010000    (当前)正向有功总电能
 | 
			
		||||
            
 | 
			
		||||
            """;
 | 
			
		||||
        return base.GetAddressDescription() + Environment.NewLine + str;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<T> LoadSourceRead<T, T2>(List<T2> deviceVariables, int maxPack)
 | 
			
		||||
    {
 | 
			
		||||
        return PackHelper.LoadSourceRead<T, T2>(this, deviceVariables, maxPack);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult<byte[]> Read(string address, int length, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            Connect(cancellationToken);
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command(address, (byte)ControlCode.Read, Station);
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = WaitingClientEx.SendThenResponse(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                return (MessageBase)result.RequestInfo;
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult<byte[]>(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<byte[]>(ex);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task<OperResult<byte[]>> ReadAsync(string address, int length, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command(address, (byte)ControlCode.Read, Station);
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                return (MessageBase)result.RequestInfo;
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult<byte[]>(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<byte[]>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void SetDataAdapter(object socketClient = null)
 | 
			
		||||
    {
 | 
			
		||||
        var dataHandleAdapter = new DLT645_2007DataHandleAdapter
 | 
			
		||||
        {
 | 
			
		||||
            EnableFEHead = EnableFEHead
 | 
			
		||||
        };
 | 
			
		||||
        SerialSession.SetDataHandlingAdapter(dataHandleAdapter);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult Write(string address, string value, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            Connect(cancellationToken);
 | 
			
		||||
            Password ??= string.Empty;
 | 
			
		||||
            OperCode ??= string.Empty;
 | 
			
		||||
            if (Password.Length < 8)
 | 
			
		||||
                Password = Password.PadLeft(8, '0');
 | 
			
		||||
            if (OperCode.Length < 8)
 | 
			
		||||
                OperCode = OperCode.PadLeft(8, '0');
 | 
			
		||||
            var data = DataTransUtil.SpliceArray(Password.ByHexStringToBytes(), OperCode.ByHexStringToBytes());
 | 
			
		||||
            string[] strArray = value.Split(new string[] { ";" }, StringSplitOptions.RemoveEmptyEntries);
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command(address, (byte)ControlCode.Write, Station, data, strArray);
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = WaitingClientEx.SendThenResponse(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                return (MessageBase)result.RequestInfo;
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<byte[]>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult Write(string address, byte[] value, CancellationToken cancellationToken = default) => Write(address, value.ToString(), cancellationToken);
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult Write(string address, bool[] value, CancellationToken cancellationToken = default) => Write(address, value.ToString(), cancellationToken);
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task<OperResult> WriteAsync(string address, string value, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
            Password ??= string.Empty;
 | 
			
		||||
            OperCode ??= string.Empty;
 | 
			
		||||
            if (Password.Length < 8)
 | 
			
		||||
                Password = Password.PadLeft(8, '0');
 | 
			
		||||
            if (OperCode.Length < 8)
 | 
			
		||||
                OperCode = OperCode.PadLeft(8, '0');
 | 
			
		||||
            var data = DataTransUtil.SpliceArray(Password.ByHexStringToBytes(), OperCode.ByHexStringToBytes());
 | 
			
		||||
            string[] strArray = value.Split(new string[] { ";" }, StringSplitOptions.RemoveEmptyEntries);
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command(address, (byte)ControlCode.Write, Station, data, strArray);
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                return (MessageBase)result.RequestInfo;
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<byte[]>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, byte[] value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, uint value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, double value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, float value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, long value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, ulong value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, ushort value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, short value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, int value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, bool[] value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    #region 其他方法
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 广播校时
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="dateTime"></param>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public OperResult BroadcastTime(DateTime dateTime, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            Connect(cancellationToken);
 | 
			
		||||
            string str = $"{dateTime.Second:D2}{dateTime.Minute:D2}{dateTime.Hour:D2}{dateTime.Day:D2}{dateTime.Month:D2}{dateTime.Year % 100:D2}";
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command((byte)ControlCode.BroadcastTime, str.ByHexStringToBytes().ToArray(), "999999999999".ByHexStringToBytes());
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                SerialSession.Send(commandResult.Content);
 | 
			
		||||
                return OperResult.CreateSuccessResult();
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<string>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 冻结
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="dateTime"></param>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public async Task<OperResult> FreezeAsync(DateTime dateTime, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
            string str = $"{dateTime.Minute:D2}{dateTime.Hour:D2}{dateTime.Day:D2}{dateTime.Month:D2}";
 | 
			
		||||
            if (Station.IsNullOrEmpty()) Station = string.Empty;
 | 
			
		||||
            if (Station.Length < 12) Station = Station.PadLeft(12, '0');
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command((byte)ControlCode.Freeze, str.ByHexStringToBytes().ToArray(), Station.ByHexStringToBytes().Reverse().ToArray());
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                var result1 = ((MessageBase)result.RequestInfo);
 | 
			
		||||
                if (result1.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult(result1);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<string>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 读取通信地址
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public async Task<OperResult<string>> ReadDeviceStationAsync(CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command((byte)ControlCode.ReadStation, null, "AAAAAAAAAAAA".ByHexStringToBytes());
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                var result1 = ((MessageBase)result.RequestInfo);
 | 
			
		||||
                if (result1.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    var buffer = result1.Content.SelectMiddle(0, 6).BytesAdd(-0x33);
 | 
			
		||||
                    return OperResult.CreateSuccessResult(buffer.Reverse().ToArray().ToHexString());
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult<string>(result1);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult<string>(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<string>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 修改波特率
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="baudRate"></param>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public async Task<OperResult> WriteBaudRateAsync(int baudRate, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
            byte baudRateByte;
 | 
			
		||||
            switch (baudRate)
 | 
			
		||||
            {
 | 
			
		||||
                case 600: baudRateByte = 0x02; break;
 | 
			
		||||
                case 1200: baudRateByte = 0x04; break;
 | 
			
		||||
                case 2400: baudRateByte = 0x08; break;
 | 
			
		||||
                case 4800: baudRateByte = 0x10; break;
 | 
			
		||||
                case 9600: baudRateByte = 0x20; break;
 | 
			
		||||
                case 19200: baudRateByte = 0x40; break;
 | 
			
		||||
                default: return new OperResult<string>($"不支持此波特率:{baudRate}");
 | 
			
		||||
            }
 | 
			
		||||
            if (Station.IsNullOrEmpty()) Station = string.Empty;
 | 
			
		||||
            if (Station.Length < 12) Station = Station.PadLeft(12, '0');
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command((byte)ControlCode.WriteBaudRate, new byte[] { baudRateByte }, Station.ByHexStringToBytes().Reverse().ToArray());
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                var result1 = ((MessageBase)result.RequestInfo);
 | 
			
		||||
                if (result1.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult(result1);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<string>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 更新通信地址
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="station"></param>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public async Task<OperResult> WriteDeviceStationAsync(string station, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command((byte)ControlCode.WriteStation, station.ByHexStringToBytes().Reverse().ToArray(), "AAAAAAAAAAAA".ByHexStringToBytes());
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                var result1 = ((MessageBase)result.RequestInfo);
 | 
			
		||||
                if (result1.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult(result1);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<string>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 修改密码
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="level"></param>
 | 
			
		||||
    /// <param name="oldPassword"></param>
 | 
			
		||||
    /// <param name="newPassword"></param>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public async Task<OperResult> WritePasswordAsync(byte level, string oldPassword, string newPassword, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
 | 
			
		||||
            if (Station.IsNullOrEmpty()) Station = string.Empty;
 | 
			
		||||
            if (Station.Length < 12) Station = Station.PadLeft(12, '0');
 | 
			
		||||
            string str = $"04000C{(level + 1):D2}";
 | 
			
		||||
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command((byte)ControlCode.WritePassword,
 | 
			
		||||
                str.ByHexStringToBytes().Reverse().ToArray()
 | 
			
		||||
                .SpliceArray(oldPassword.ByHexStringToBytes().Reverse().ToArray())
 | 
			
		||||
                .SpliceArray(newPassword.ByHexStringToBytes().Reverse().ToArray())
 | 
			
		||||
                , Station.ByHexStringToBytes().Reverse().ToArray());
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                var result1 = ((MessageBase)result.RequestInfo);
 | 
			
		||||
                if (result1.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult(result1);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<string>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    #endregion
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,480 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.ComponentModel;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Extension.Generic;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Adapter.DLT645;
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// DLT645_2007
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class DLT645_2007OverTcp : ReadWriteDevicesTcpClientBase
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// DLT645_2007
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="tcpClient"></param>
 | 
			
		||||
    public DLT645_2007OverTcp(TcpClient tcpClient) : base(tcpClient)
 | 
			
		||||
    {
 | 
			
		||||
        ThingsGatewayBitConverter = new DLT645_2007BitConverter(EndianType.Big);
 | 
			
		||||
        RegisterByteLength = 2;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 增加FE FE FE FE的报文头部
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("前导符报文头")]
 | 
			
		||||
    public bool EnableFEHead { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 写入需操作员代码
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("操作员代码")]
 | 
			
		||||
    public string OperCode { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 写入密码
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("写入密码")]
 | 
			
		||||
    public string Password { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 通讯地址BCD码,一般应该是12个字符
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("通讯地址")]
 | 
			
		||||
    public string Station { get; set; }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override string GetAddressDescription()
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
        var str = """
 | 
			
		||||
            查看附带文档或者相关资料,下面列举一下常见的数据标识地址 
 | 
			
		||||
            
 | 
			
		||||
            地址                       说明                    
 | 
			
		||||
            -----------------------------------------
 | 
			
		||||
            02010100    A相电压
 | 
			
		||||
            02020100    A相电流
 | 
			
		||||
            02030000    瞬时总有功功率
 | 
			
		||||
            00000000    (当前)组合有功总电能
 | 
			
		||||
            00010000    (当前)正向有功总电能
 | 
			
		||||
            
 | 
			
		||||
            """;
 | 
			
		||||
        return base.GetAddressDescription() + Environment.NewLine + str;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<T> LoadSourceRead<T, T2>(List<T2> deviceVariables, int maxPack)
 | 
			
		||||
    {
 | 
			
		||||
        return PackHelper.LoadSourceRead<T, T2>(this, deviceVariables, maxPack);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult<byte[]> Read(string address, int length, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            Connect(cancellationToken);
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command(address, (byte)ControlCode.Read, Station);
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = WaitingClientEx.SendThenResponse(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                return (MessageBase)result.RequestInfo;
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult<byte[]>(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<byte[]>(ex);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task<OperResult<byte[]>> ReadAsync(string address, int length, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command(address, (byte)ControlCode.Read, Station);
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                return (MessageBase)result.RequestInfo;
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult<byte[]>(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<byte[]>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void SetDataAdapter(object socketClient = null)
 | 
			
		||||
    {
 | 
			
		||||
        var dataHandleAdapter = new DLT645_2007DataHandleAdapter
 | 
			
		||||
        {
 | 
			
		||||
            EnableFEHead = EnableFEHead
 | 
			
		||||
        };
 | 
			
		||||
        TcpClient.SetDataHandlingAdapter(dataHandleAdapter);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult Write(string address, string value, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            Connect(cancellationToken);
 | 
			
		||||
            Password ??= string.Empty;
 | 
			
		||||
            OperCode ??= string.Empty;
 | 
			
		||||
            if (Password.Length < 8)
 | 
			
		||||
                Password = Password.PadLeft(8, '0');
 | 
			
		||||
            if (OperCode.Length < 8)
 | 
			
		||||
                OperCode = OperCode.PadLeft(8, '0');
 | 
			
		||||
            var data = DataTransUtil.SpliceArray(Password.ByHexStringToBytes(), OperCode.ByHexStringToBytes());
 | 
			
		||||
            string[] strArray = value.Split(new string[] { ";" }, StringSplitOptions.RemoveEmptyEntries);
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command(address, (byte)ControlCode.Write, Station, data, strArray);
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = WaitingClientEx.SendThenResponse(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                return (MessageBase)result.RequestInfo;
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<byte[]>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult Write(string address, byte[] value, CancellationToken cancellationToken = default) => Write(address, value.ToString(), cancellationToken);
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult Write(string address, bool[] value, CancellationToken cancellationToken = default) => Write(address, value.ToString(), cancellationToken);
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task<OperResult> WriteAsync(string address, string value, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
            Password ??= string.Empty;
 | 
			
		||||
            OperCode ??= string.Empty;
 | 
			
		||||
            if (Password.Length < 8)
 | 
			
		||||
                Password = Password.PadLeft(8, '0');
 | 
			
		||||
            if (OperCode.Length < 8)
 | 
			
		||||
                OperCode = OperCode.PadLeft(8, '0');
 | 
			
		||||
            var data = DataTransUtil.SpliceArray(Password.ByHexStringToBytes(), OperCode.ByHexStringToBytes());
 | 
			
		||||
            string[] strArray = value.Split(new string[] { ";" }, StringSplitOptions.RemoveEmptyEntries);
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command(address, (byte)ControlCode.Write, Station, data, strArray);
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                return (MessageBase)result.RequestInfo;
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<byte[]>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, byte[] value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, uint value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, double value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, float value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, long value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, ulong value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, ushort value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, short value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, int value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, bool[] value, CancellationToken cancellationToken = default) => WriteAsync(address, value.ToString(), cancellationToken);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    #region 其他方法
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 广播校时
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="dateTime"></param>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public OperResult BroadcastTime(DateTime dateTime, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            Connect(cancellationToken);
 | 
			
		||||
            string str = $"{dateTime.Second:D2}{dateTime.Minute:D2}{dateTime.Hour:D2}{dateTime.Day:D2}{dateTime.Month:D2}{dateTime.Year % 100:D2}";
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command((byte)ControlCode.BroadcastTime, str.ByHexStringToBytes().Reverse().ToArray(), "999999999999".ByHexStringToBytes());
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                TcpClient.Send(commandResult.Content);
 | 
			
		||||
                return OperResult.CreateSuccessResult();
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<string>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 冻结
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="dateTime"></param>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public async Task<OperResult> FreezeAsync(DateTime dateTime, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
            string str = $"{dateTime.Minute:D2}{dateTime.Hour:D2}{dateTime.Day:D2}{dateTime.Month:D2}";
 | 
			
		||||
            if (Station.IsNullOrEmpty()) Station = string.Empty;
 | 
			
		||||
            if (Station.Length < 12) Station = Station.PadLeft(12, '0');
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command((byte)ControlCode.Freeze, str.ByHexStringToBytes().Reverse().ToArray(), Station.ByHexStringToBytes().Reverse().ToArray());
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                var result1 = ((MessageBase)result.RequestInfo);
 | 
			
		||||
                if (result1.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult(result1);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<string>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 读取通信地址
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public async Task<OperResult<string>> ReadDeviceStationAsync(CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command((byte)ControlCode.ReadStation, null, "AAAAAAAAAAAA".ByHexStringToBytes());
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                var result1 = ((MessageBase)result.RequestInfo);
 | 
			
		||||
                if (result1.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    var buffer = result1.Content.SelectMiddle(0, 6).BytesAdd(-0x33);
 | 
			
		||||
                    return OperResult.CreateSuccessResult(buffer.Reverse().ToArray().ToHexString());
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult<string>(result1);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult<string>(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<string>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 修改波特率
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="baudRate"></param>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public async Task<OperResult> WriteBaudRateAsync(int baudRate, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
            byte baudRateByte;
 | 
			
		||||
            switch (baudRate)
 | 
			
		||||
            {
 | 
			
		||||
                case 600: baudRateByte = 0x02; break;
 | 
			
		||||
                case 1200: baudRateByte = 0x04; break;
 | 
			
		||||
                case 2400: baudRateByte = 0x08; break;
 | 
			
		||||
                case 4800: baudRateByte = 0x10; break;
 | 
			
		||||
                case 9600: baudRateByte = 0x20; break;
 | 
			
		||||
                case 19200: baudRateByte = 0x40; break;
 | 
			
		||||
                default: return new OperResult<string>($"不支持此波特率:{baudRate}");
 | 
			
		||||
            }
 | 
			
		||||
            if (Station.IsNullOrEmpty()) Station = string.Empty;
 | 
			
		||||
            if (Station.Length < 12) Station = Station.PadLeft(12, '0');
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command((byte)ControlCode.WriteBaudRate, new byte[] { baudRateByte }, Station.ByHexStringToBytes().Reverse().ToArray());
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                var result1 = ((MessageBase)result.RequestInfo);
 | 
			
		||||
                if (result1.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult(result1);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<string>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 更新通信地址
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="station"></param>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public async Task<OperResult> WriteDeviceStationAsync(string station, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command((byte)ControlCode.WriteStation, station.ByHexStringToBytes().Reverse().ToArray(), "AAAAAAAAAAAA".ByHexStringToBytes());
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                var result1 = ((MessageBase)result.RequestInfo);
 | 
			
		||||
                if (result1.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult(result1);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<string>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 修改密码
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="level"></param>
 | 
			
		||||
    /// <param name="oldPassword"></param>
 | 
			
		||||
    /// <param name="newPassword"></param>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public async Task<OperResult> WritePasswordAsync(byte level, string oldPassword, string newPassword, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await ConnectAsync(cancellationToken);
 | 
			
		||||
 | 
			
		||||
            if (Station.IsNullOrEmpty()) Station = string.Empty;
 | 
			
		||||
            if (Station.Length < 12) Station = Station.PadLeft(12, '0');
 | 
			
		||||
            string str = $"04000C{level:D2}";
 | 
			
		||||
 | 
			
		||||
            var commandResult = DLT645Helper.GetDLT645_2007Command((byte)ControlCode.WritePassword,
 | 
			
		||||
                str.ByHexStringToBytes().Reverse().ToArray()
 | 
			
		||||
                .SpliceArray(oldPassword.ByHexStringToBytes().Reverse().ToArray())
 | 
			
		||||
                .SpliceArray(newPassword.ByHexStringToBytes().Reverse().ToArray())
 | 
			
		||||
                , Station.ByHexStringToBytes().Reverse().ToArray());
 | 
			
		||||
            if (commandResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                var result = await WaitingClientEx.SendThenResponseAsync(commandResult.Content, TimeOut, cancellationToken);
 | 
			
		||||
                var result1 = ((MessageBase)result.RequestInfo);
 | 
			
		||||
                if (result1.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult(result1);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(commandResult);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<string>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    #endregion
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,421 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.Collections.Concurrent;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Extension.Bool;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.Generic;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Adapter.Modbus;
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class ModbusSerialServer : ReadWriteDevicesSerialSessionBase
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 接收外部写入时,传出变量地址/写入字节组/转换规则/客户端
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public Func<ModbusAddress, byte[], IThingsGatewayBitConverter, SerialSession, Task<OperResult>> WriteData;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 继电器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private ConcurrentDictionary<byte, ByteBlock> ModbusServer01ByteBlocks = new();
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 开关输入
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private ConcurrentDictionary<byte, ByteBlock> ModbusServer02ByteBlocks = new();
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 输入寄存器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private ConcurrentDictionary<byte, ByteBlock> ModbusServer03ByteBlocks = new();
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 保持寄存器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private ConcurrentDictionary<byte, ByteBlock> ModbusServer04ByteBlocks = new();
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public ModbusSerialServer(SerialSession serialSession) : base(serialSession)
 | 
			
		||||
    {
 | 
			
		||||
        ThingsGatewayBitConverter = new ThingsGatewayBitConverter(EndianType.Big);
 | 
			
		||||
        RegisterByteLength = 2;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 多站点
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool MulStation { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 默认站点
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public byte Station { get; set; } = 1;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void Dispose()
 | 
			
		||||
    {
 | 
			
		||||
        foreach (var item in ModbusServer01ByteBlocks)
 | 
			
		||||
        {
 | 
			
		||||
            item.Value.SafeDispose();
 | 
			
		||||
        }
 | 
			
		||||
        foreach (var item in ModbusServer02ByteBlocks)
 | 
			
		||||
        {
 | 
			
		||||
            item.Value.SafeDispose();
 | 
			
		||||
        }
 | 
			
		||||
        foreach (var item in ModbusServer03ByteBlocks)
 | 
			
		||||
        {
 | 
			
		||||
            item.Value.SafeDispose();
 | 
			
		||||
        }
 | 
			
		||||
        foreach (var item in ModbusServer04ByteBlocks)
 | 
			
		||||
        {
 | 
			
		||||
            item.Value.SafeDispose();
 | 
			
		||||
        }
 | 
			
		||||
        ModbusServer01ByteBlocks.Clear();
 | 
			
		||||
        ModbusServer02ByteBlocks.Clear();
 | 
			
		||||
        ModbusServer03ByteBlocks.Clear();
 | 
			
		||||
        ModbusServer04ByteBlocks.Clear();
 | 
			
		||||
        base.Dispose();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override string GetAddressDescription()
 | 
			
		||||
    {
 | 
			
		||||
        return base.GetAddressDescription() + Environment.NewLine + ModbusHelper.GetAddressDescription();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<T> LoadSourceRead<T, T2>(List<T2> deviceVariables, int maxPack)
 | 
			
		||||
    {
 | 
			
		||||
        return PackHelper.LoadSourceRead<T, T2>(this, deviceVariables, maxPack);
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult<byte[]> Read(string address, int length, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        ModbusAddress mAddress;
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            mAddress = ModbusAddress.ParseFrom(address, Station);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<byte[]>(ex);
 | 
			
		||||
        }
 | 
			
		||||
        if (MulStation)
 | 
			
		||||
        {
 | 
			
		||||
            Init(mAddress);
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            if (Station != mAddress.Station)
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult<byte[]>("地址错误");
 | 
			
		||||
            }
 | 
			
		||||
            Init(mAddress);
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        var ModbusServer01ByteBlock = ModbusServer01ByteBlocks[mAddress.Station];
 | 
			
		||||
        var ModbusServer02ByteBlock = ModbusServer02ByteBlocks[mAddress.Station];
 | 
			
		||||
        var ModbusServer03ByteBlock = ModbusServer03ByteBlocks[mAddress.Station];
 | 
			
		||||
        var ModbusServer04ByteBlock = ModbusServer04ByteBlocks[mAddress.Station];
 | 
			
		||||
        int len = mAddress.ReadFunction == 2 || mAddress.ReadFunction == 1 ? length : length * RegisterByteLength;
 | 
			
		||||
        switch (mAddress.ReadFunction)
 | 
			
		||||
        {
 | 
			
		||||
            case 1:
 | 
			
		||||
                byte[] bytes0 = new byte[len];
 | 
			
		||||
                ModbusServer01ByteBlock.Pos = mAddress.AddressStart;
 | 
			
		||||
                ModbusServer01ByteBlock.Read(bytes0);
 | 
			
		||||
                return OperResult.CreateSuccessResult(bytes0);
 | 
			
		||||
            case 2:
 | 
			
		||||
                byte[] bytes1 = new byte[len];
 | 
			
		||||
                ModbusServer02ByteBlock.Pos = mAddress.AddressStart;
 | 
			
		||||
                ModbusServer02ByteBlock.Read(bytes1);
 | 
			
		||||
                return OperResult.CreateSuccessResult(bytes1);
 | 
			
		||||
            case 3:
 | 
			
		||||
 | 
			
		||||
                byte[] bytes3 = new byte[len];
 | 
			
		||||
                ModbusServer03ByteBlock.Pos = mAddress.AddressStart * RegisterByteLength;
 | 
			
		||||
                ModbusServer03ByteBlock.Read(bytes3);
 | 
			
		||||
                return OperResult.CreateSuccessResult(bytes3);
 | 
			
		||||
            case 4:
 | 
			
		||||
                byte[] bytes4 = new byte[len];
 | 
			
		||||
                ModbusServer04ByteBlock.Pos = mAddress.AddressStart * RegisterByteLength;
 | 
			
		||||
                ModbusServer04ByteBlock.Read(bytes4);
 | 
			
		||||
                return OperResult.CreateSuccessResult(bytes4);
 | 
			
		||||
        }
 | 
			
		||||
        return new OperResult<byte[]>("功能码错误");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult<byte[]>> ReadAsync(string address, int length, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        return Task.FromResult(Read(address, length));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void SetDataAdapter(object socketClient)
 | 
			
		||||
    {
 | 
			
		||||
        ModbusSerialServerDataHandleAdapter dataHandleAdapter = new();
 | 
			
		||||
        dataHandleAdapter.CacheTimeout = TimeSpan.FromMilliseconds(CacheTimeout);
 | 
			
		||||
        SerialSession.SetDataHandlingAdapter(dataHandleAdapter);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult Write(string address, byte[] value, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        ModbusAddress mAddress;
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            mAddress = ModbusAddress.ParseFrom(address, Station);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult(ex);
 | 
			
		||||
        }
 | 
			
		||||
        if (MulStation)
 | 
			
		||||
        {
 | 
			
		||||
            Init(mAddress);
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            if (Station != mAddress.Station)
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult("地址错误");
 | 
			
		||||
            }
 | 
			
		||||
            Init(mAddress);
 | 
			
		||||
        }
 | 
			
		||||
        var ModbusServer03ByteBlock = ModbusServer03ByteBlocks[mAddress.Station];
 | 
			
		||||
        var ModbusServer04ByteBlock = ModbusServer04ByteBlocks[mAddress.Station];
 | 
			
		||||
        switch (mAddress.ReadFunction)
 | 
			
		||||
        {
 | 
			
		||||
            case 3:
 | 
			
		||||
                ModbusServer03ByteBlock.Pos = mAddress.AddressStart * RegisterByteLength;
 | 
			
		||||
                ModbusServer03ByteBlock.Write(value);
 | 
			
		||||
                return OperResult.CreateSuccessResult();
 | 
			
		||||
            case 4:
 | 
			
		||||
                ModbusServer04ByteBlock.Pos = mAddress.AddressStart * RegisterByteLength;
 | 
			
		||||
                ModbusServer04ByteBlock.Write(value);
 | 
			
		||||
                return OperResult.CreateSuccessResult();
 | 
			
		||||
        }
 | 
			
		||||
        return new OperResult("功能码错误");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult Write(string address, bool[] value, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        ModbusAddress mAddress;
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            mAddress = ModbusAddress.ParseFrom(address, Station);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return (new OperResult(ex));
 | 
			
		||||
        }
 | 
			
		||||
        if (MulStation)
 | 
			
		||||
        {
 | 
			
		||||
            Init(mAddress);
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            if (Station != mAddress.Station)
 | 
			
		||||
            {
 | 
			
		||||
                return (new OperResult("地址错误"));
 | 
			
		||||
            }
 | 
			
		||||
            Init(mAddress);
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        var ModbusServer01ByteBlock = ModbusServer01ByteBlocks[mAddress.Station];
 | 
			
		||||
        var ModbusServer02ByteBlock = ModbusServer02ByteBlocks[mAddress.Station];
 | 
			
		||||
        switch (mAddress.ReadFunction)
 | 
			
		||||
        {
 | 
			
		||||
            case 1:
 | 
			
		||||
                ModbusServer01ByteBlock.Pos = mAddress.AddressStart;
 | 
			
		||||
                ModbusServer01ByteBlock.Write(value.BoolArrayToByte());
 | 
			
		||||
                return (OperResult.CreateSuccessResult());
 | 
			
		||||
            case 2:
 | 
			
		||||
                ModbusServer02ByteBlock.Pos = mAddress.AddressStart;
 | 
			
		||||
                ModbusServer02ByteBlock.Write(value.BoolArrayToByte());
 | 
			
		||||
                return (OperResult.CreateSuccessResult());
 | 
			
		||||
        }
 | 
			
		||||
        return new OperResult("功能码错误");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, byte[] value, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        return Task.FromResult(Write(address, value));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, bool[] value, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        return Task.FromResult(Write(address, value));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override async Task Received(SerialSession client, ReceivedDataEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var requestInfo = e.RequestInfo;
 | 
			
		||||
            //接收外部报文
 | 
			
		||||
            if (requestInfo is ModbusSerialServerMessage modbusServerMessage)
 | 
			
		||||
            {
 | 
			
		||||
                if (modbusServerMessage.CurModbusAddress == null)
 | 
			
		||||
                {
 | 
			
		||||
                    return;//无法解析直接返回
 | 
			
		||||
                }
 | 
			
		||||
                if (!modbusServerMessage.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    return;//无法解析直接返回
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                if (modbusServerMessage.CurModbusAddress.WriteFunction == 0)//读取
 | 
			
		||||
                {
 | 
			
		||||
                    var data = Read(modbusServerMessage.CurModbusAddress.ToString(), modbusServerMessage.Length);
 | 
			
		||||
                    if (data.IsSuccess)
 | 
			
		||||
                    {
 | 
			
		||||
                        var coreData = data.Content;
 | 
			
		||||
                        if (modbusServerMessage.CurModbusAddress.ReadFunction == 1 || modbusServerMessage.CurModbusAddress.ReadFunction == 2)
 | 
			
		||||
                        {
 | 
			
		||||
                            coreData = data.Content.Select(m => m > 0).ToArray().BoolArrayToByte().SelectMiddle(0, (int)Math.Ceiling(modbusServerMessage.Length / 8.0));
 | 
			
		||||
                        }
 | 
			
		||||
                        var sendData = modbusServerMessage.ReceivedBytes.SelectMiddle(0, 2).SpliceArray(new byte[] { (byte)coreData.Length }, coreData);
 | 
			
		||||
                        SerialSession.Send(sendData);
 | 
			
		||||
                    }
 | 
			
		||||
                    else
 | 
			
		||||
                    {
 | 
			
		||||
                        WriteError(SerialSession, modbusServerMessage);//返回错误码
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                else//写入
 | 
			
		||||
                {
 | 
			
		||||
                    var coreData = modbusServerMessage.Content;
 | 
			
		||||
                    if (modbusServerMessage.CurModbusAddress.ReadFunction == 1 || modbusServerMessage.CurModbusAddress.ReadFunction == 2)
 | 
			
		||||
                    {
 | 
			
		||||
                        //写入继电器
 | 
			
		||||
                        if (WriteData != null)
 | 
			
		||||
                        {
 | 
			
		||||
                            // 接收外部写入时,传出变量地址/写入字节组/转换规则/客户端
 | 
			
		||||
                            if ((await WriteData(modbusServerMessage.CurModbusAddress, modbusServerMessage.Content, ThingsGatewayBitConverter, SerialSession)).IsSuccess)
 | 
			
		||||
                            {
 | 
			
		||||
                                var result = Write(modbusServerMessage.CurModbusAddress.ToString(), coreData.ByteToBoolArray(modbusServerMessage.Length));
 | 
			
		||||
                                if (result.IsSuccess)
 | 
			
		||||
                                {
 | 
			
		||||
                                    WriteSuccess03(SerialSession, modbusServerMessage);
 | 
			
		||||
                                }
 | 
			
		||||
                                else
 | 
			
		||||
                                {
 | 
			
		||||
                                    WriteError(SerialSession, modbusServerMessage);
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                WriteError(SerialSession, modbusServerMessage);
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        else
 | 
			
		||||
                        {
 | 
			
		||||
                            //写入内存区
 | 
			
		||||
                            var result = Write(modbusServerMessage.CurModbusAddress.ToString(), coreData.ByteToBoolArray(modbusServerMessage.Length));
 | 
			
		||||
                            if (result.IsSuccess)
 | 
			
		||||
                            {
 | 
			
		||||
                                WriteSuccess03(SerialSession, modbusServerMessage);
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                WriteError(SerialSession, modbusServerMessage);
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    else
 | 
			
		||||
                    {
 | 
			
		||||
                        //写入寄存器
 | 
			
		||||
                        if (WriteData != null)
 | 
			
		||||
                        {
 | 
			
		||||
 | 
			
		||||
                            if ((await WriteData(modbusServerMessage.CurModbusAddress, modbusServerMessage.Content, ThingsGatewayBitConverter, SerialSession)).IsSuccess)
 | 
			
		||||
                            {
 | 
			
		||||
                                var result = Write(modbusServerMessage.CurModbusAddress.ToString(), coreData);
 | 
			
		||||
                                if (result.IsSuccess)
 | 
			
		||||
                                {
 | 
			
		||||
                                    WriteSuccess03(SerialSession, modbusServerMessage);
 | 
			
		||||
                                }
 | 
			
		||||
                                else
 | 
			
		||||
                                {
 | 
			
		||||
                                    WriteError(SerialSession, modbusServerMessage);
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                WriteError(SerialSession, modbusServerMessage);
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        else
 | 
			
		||||
                        {
 | 
			
		||||
                            var result = Write(modbusServerMessage.CurModbusAddress.ToString(), coreData);
 | 
			
		||||
                            if (result.IsSuccess)
 | 
			
		||||
                            {
 | 
			
		||||
                                WriteSuccess03(SerialSession, modbusServerMessage);
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                WriteError(SerialSession, modbusServerMessage);
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            Logger.LogError(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
        //返回错误码
 | 
			
		||||
        static void WriteError(SerialSession SerialSession, ModbusSerialServerMessage modbusServerMessage)
 | 
			
		||||
        {
 | 
			
		||||
            var sendData = modbusServerMessage.ReceivedBytes.SelectMiddle(0, 2)
 | 
			
		||||
.SpliceArray(new byte[] { (byte)1 });//01 lllegal function
 | 
			
		||||
            sendData[1] = (byte)(sendData[1] + 128);
 | 
			
		||||
            SerialSession.Send(sendData);
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private static void WriteSuccess03(SerialSession SerialSession, ModbusSerialServerMessage modbusServerMessage)
 | 
			
		||||
    {
 | 
			
		||||
        var sendData = modbusServerMessage.ReceivedBytes.SelectMiddle(0, 6);
 | 
			
		||||
        SerialSession.Send(sendData);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void Init(ModbusAddress mAddress)
 | 
			
		||||
    {
 | 
			
		||||
        if (ModbusServer01ByteBlocks.TryAdd(mAddress.Station, new(1024 * 128)))
 | 
			
		||||
            ModbusServer01ByteBlocks[mAddress.Station].SetLength(1024 * 128);
 | 
			
		||||
        if (ModbusServer02ByteBlocks.TryAdd(mAddress.Station, new(1024 * 128)))
 | 
			
		||||
            ModbusServer02ByteBlocks[mAddress.Station].SetLength(1024 * 128);
 | 
			
		||||
        if (ModbusServer03ByteBlocks.TryAdd(mAddress.Station, new(1024 * 128)))
 | 
			
		||||
            ModbusServer03ByteBlocks[mAddress.Station].SetLength(1024 * 128);
 | 
			
		||||
        if (ModbusServer04ByteBlocks.TryAdd(mAddress.Station, new(1024 * 128)))
 | 
			
		||||
            ModbusServer04ByteBlocks[mAddress.Station].SetLength(1024 * 128);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,441 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.Collections.Concurrent;
 | 
			
		||||
using System.ComponentModel;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Extension.Bool;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.Generic;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Adapter.Modbus;
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class ModbusTcpServer : ReadWriteDevicesTcpServerBase
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 接收外部写入时,传出变量地址/写入字节组/转换规则/客户端
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public Func<ModbusAddress, byte[], IThingsGatewayBitConverter, SocketClient, Task<OperResult>> WriteData;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 继电器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private ConcurrentDictionary<byte, ByteBlock> ModbusServer01ByteBlocks = new();
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 开关输入
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private ConcurrentDictionary<byte, ByteBlock> ModbusServer02ByteBlocks = new();
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 输入寄存器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private ConcurrentDictionary<byte, ByteBlock> ModbusServer03ByteBlocks = new();
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 保持寄存器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private ConcurrentDictionary<byte, ByteBlock> ModbusServer04ByteBlocks = new();
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public ModbusTcpServer(TcpService tcpService) : base(tcpService)
 | 
			
		||||
    {
 | 
			
		||||
        ThingsGatewayBitConverter = new ThingsGatewayBitConverter(EndianType.Big);
 | 
			
		||||
        RegisterByteLength = 2;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 多站点
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("多站点")]
 | 
			
		||||
    public bool MulStation { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 默认站点
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("默认站点")]
 | 
			
		||||
    public byte Station { get; set; } = 1;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void Dispose()
 | 
			
		||||
    {
 | 
			
		||||
        foreach (var item in ModbusServer01ByteBlocks)
 | 
			
		||||
        {
 | 
			
		||||
            item.Value.SafeDispose();
 | 
			
		||||
        }
 | 
			
		||||
        foreach (var item in ModbusServer02ByteBlocks)
 | 
			
		||||
        {
 | 
			
		||||
            item.Value.SafeDispose();
 | 
			
		||||
        }
 | 
			
		||||
        foreach (var item in ModbusServer03ByteBlocks)
 | 
			
		||||
        {
 | 
			
		||||
            item.Value.SafeDispose();
 | 
			
		||||
        }
 | 
			
		||||
        foreach (var item in ModbusServer04ByteBlocks)
 | 
			
		||||
        {
 | 
			
		||||
            item.Value.SafeDispose();
 | 
			
		||||
        }
 | 
			
		||||
        ModbusServer01ByteBlocks.Clear();
 | 
			
		||||
        ModbusServer02ByteBlocks.Clear();
 | 
			
		||||
        ModbusServer03ByteBlocks.Clear();
 | 
			
		||||
        ModbusServer04ByteBlocks.Clear();
 | 
			
		||||
        base.Dispose();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override string GetAddressDescription()
 | 
			
		||||
    {
 | 
			
		||||
        return base.GetAddressDescription() + Environment.NewLine + ModbusHelper.GetAddressDescription();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<T> LoadSourceRead<T, T2>(List<T2> deviceVariables, int maxPack)
 | 
			
		||||
    {
 | 
			
		||||
        return PackHelper.LoadSourceRead<T, T2>(this, deviceVariables, maxPack);
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult<byte[]> Read(string address, int length, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        ModbusAddress mAddress;
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            mAddress = ModbusAddress.ParseFrom(address, Station);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<byte[]>(ex);
 | 
			
		||||
        }
 | 
			
		||||
        if (MulStation)
 | 
			
		||||
        {
 | 
			
		||||
            Init(mAddress);
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            if (Station != mAddress.Station)
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult<byte[]>("地址错误");
 | 
			
		||||
            }
 | 
			
		||||
            Init(mAddress);
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        var ModbusServer01ByteBlock = ModbusServer01ByteBlocks[mAddress.Station];
 | 
			
		||||
        var ModbusServer02ByteBlock = ModbusServer02ByteBlocks[mAddress.Station];
 | 
			
		||||
        var ModbusServer03ByteBlock = ModbusServer03ByteBlocks[mAddress.Station];
 | 
			
		||||
        var ModbusServer04ByteBlock = ModbusServer04ByteBlocks[mAddress.Station];
 | 
			
		||||
        int len = mAddress.ReadFunction == 2 || mAddress.ReadFunction == 1 ? length : length * RegisterByteLength;
 | 
			
		||||
        switch (mAddress.ReadFunction)
 | 
			
		||||
        {
 | 
			
		||||
            case 1:
 | 
			
		||||
                byte[] bytes0 = new byte[len];
 | 
			
		||||
                ModbusServer01ByteBlock.Pos = mAddress.AddressStart;
 | 
			
		||||
                ModbusServer01ByteBlock.Read(bytes0);
 | 
			
		||||
                return OperResult.CreateSuccessResult(bytes0);
 | 
			
		||||
            case 2:
 | 
			
		||||
                byte[] bytes1 = new byte[len];
 | 
			
		||||
                ModbusServer02ByteBlock.Pos = mAddress.AddressStart;
 | 
			
		||||
                ModbusServer02ByteBlock.Read(bytes1);
 | 
			
		||||
                return OperResult.CreateSuccessResult(bytes1);
 | 
			
		||||
            case 3:
 | 
			
		||||
 | 
			
		||||
                byte[] bytes3 = new byte[len];
 | 
			
		||||
                ModbusServer03ByteBlock.Pos = mAddress.AddressStart * RegisterByteLength;
 | 
			
		||||
                ModbusServer03ByteBlock.Read(bytes3);
 | 
			
		||||
                return OperResult.CreateSuccessResult(bytes3);
 | 
			
		||||
            case 4:
 | 
			
		||||
                byte[] bytes4 = new byte[len];
 | 
			
		||||
                ModbusServer04ByteBlock.Pos = mAddress.AddressStart * RegisterByteLength;
 | 
			
		||||
                ModbusServer04ByteBlock.Read(bytes4);
 | 
			
		||||
                return OperResult.CreateSuccessResult(bytes4);
 | 
			
		||||
        }
 | 
			
		||||
        return new OperResult<byte[]>("功能码错误");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult<byte[]>> ReadAsync(string address, int length, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        return Task.FromResult(Read(address, length));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void SetDataAdapter(object socketClient)
 | 
			
		||||
    {
 | 
			
		||||
        if (socketClient is SocketClient client)
 | 
			
		||||
        {
 | 
			
		||||
            ModbusTcpServerDataHandleAdapter dataHandleAdapter = new();
 | 
			
		||||
            dataHandleAdapter.CacheTimeout = TimeSpan.FromMilliseconds(CacheTimeout);
 | 
			
		||||
            client.SetDataHandlingAdapter(dataHandleAdapter);
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            foreach (var item in TcpService.GetClients())
 | 
			
		||||
            {
 | 
			
		||||
                ModbusTcpDataHandleAdapter dataHandleAdapter = new()
 | 
			
		||||
                {
 | 
			
		||||
                    CacheTimeout = TimeSpan.FromMilliseconds(CacheTimeout)
 | 
			
		||||
                };
 | 
			
		||||
                item.SetDataHandlingAdapter(dataHandleAdapter);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult Write(string address, byte[] value, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        ModbusAddress mAddress;
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            mAddress = ModbusAddress.ParseFrom(address, Station);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult(ex);
 | 
			
		||||
        }
 | 
			
		||||
        if (MulStation)
 | 
			
		||||
        {
 | 
			
		||||
            Init(mAddress);
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            if (Station != mAddress.Station)
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult("地址错误");
 | 
			
		||||
            }
 | 
			
		||||
            Init(mAddress);
 | 
			
		||||
        }
 | 
			
		||||
        var ModbusServer03ByteBlock = ModbusServer03ByteBlocks[mAddress.Station];
 | 
			
		||||
        var ModbusServer04ByteBlock = ModbusServer04ByteBlocks[mAddress.Station];
 | 
			
		||||
        switch (mAddress.ReadFunction)
 | 
			
		||||
        {
 | 
			
		||||
            case 3:
 | 
			
		||||
                ModbusServer03ByteBlock.Pos = mAddress.AddressStart * RegisterByteLength;
 | 
			
		||||
                ModbusServer03ByteBlock.Write(value);
 | 
			
		||||
                return OperResult.CreateSuccessResult();
 | 
			
		||||
            case 4:
 | 
			
		||||
                ModbusServer04ByteBlock.Pos = mAddress.AddressStart * RegisterByteLength;
 | 
			
		||||
                ModbusServer04ByteBlock.Write(value);
 | 
			
		||||
                return OperResult.CreateSuccessResult();
 | 
			
		||||
        }
 | 
			
		||||
        return new OperResult("功能码错误");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override OperResult Write(string address, bool[] value, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        ModbusAddress mAddress;
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            mAddress = ModbusAddress.ParseFrom(address, Station);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return (new OperResult(ex));
 | 
			
		||||
        }
 | 
			
		||||
        if (MulStation)
 | 
			
		||||
        {
 | 
			
		||||
            Init(mAddress);
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            if (Station != mAddress.Station)
 | 
			
		||||
            {
 | 
			
		||||
                return (new OperResult("地址错误"));
 | 
			
		||||
            }
 | 
			
		||||
            Init(mAddress);
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        var ModbusServer01ByteBlock = ModbusServer01ByteBlocks[mAddress.Station];
 | 
			
		||||
        var ModbusServer02ByteBlock = ModbusServer02ByteBlocks[mAddress.Station];
 | 
			
		||||
        switch (mAddress.ReadFunction)
 | 
			
		||||
        {
 | 
			
		||||
            case 1:
 | 
			
		||||
                ModbusServer01ByteBlock.Pos = mAddress.AddressStart;
 | 
			
		||||
                ModbusServer01ByteBlock.Write(value.BoolArrayToByte());
 | 
			
		||||
                return (OperResult.CreateSuccessResult());
 | 
			
		||||
            case 2:
 | 
			
		||||
                ModbusServer02ByteBlock.Pos = mAddress.AddressStart;
 | 
			
		||||
                ModbusServer02ByteBlock.Write(value.BoolArrayToByte());
 | 
			
		||||
                return (OperResult.CreateSuccessResult());
 | 
			
		||||
        }
 | 
			
		||||
        return new OperResult("功能码错误");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, byte[] value, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        return Task.FromResult(Write(address, value));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task<OperResult> WriteAsync(string address, bool[] value, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        return Task.FromResult(Write(address, value));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override async Task Received(SocketClient client, ReceivedDataEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var requestInfo = e.RequestInfo;
 | 
			
		||||
            //接收外部报文
 | 
			
		||||
            if (requestInfo is ModbusTcpServerMessage modbusServerMessage)
 | 
			
		||||
            {
 | 
			
		||||
                if (modbusServerMessage.CurModbusAddress == null)
 | 
			
		||||
                {
 | 
			
		||||
                    return;//无法解析直接返回
 | 
			
		||||
                }
 | 
			
		||||
                if (!modbusServerMessage.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    return;//无法解析直接返回
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                if (modbusServerMessage.CurModbusAddress.WriteFunction == 0)//读取
 | 
			
		||||
                {
 | 
			
		||||
                    var data = Read(modbusServerMessage.CurModbusAddress.ToString(), modbusServerMessage.Length);
 | 
			
		||||
                    if (data.IsSuccess)
 | 
			
		||||
                    {
 | 
			
		||||
                        var coreData = data.Content;
 | 
			
		||||
                        if (modbusServerMessage.CurModbusAddress.ReadFunction == 1 || modbusServerMessage.CurModbusAddress.ReadFunction == 2)
 | 
			
		||||
                        {
 | 
			
		||||
                            coreData = data.Content.Select(m => m > 0).ToArray().BoolArrayToByte().SelectMiddle(0, (int)Math.Ceiling(modbusServerMessage.Length / 8.0));
 | 
			
		||||
                        }
 | 
			
		||||
                        var sendData = modbusServerMessage.ReceivedBytes.SelectMiddle(0, 8).SpliceArray(new byte[] { (byte)coreData.Length }, coreData);
 | 
			
		||||
                        sendData[5] = (byte)(sendData.Length - 6);
 | 
			
		||||
                        client.Send(sendData);
 | 
			
		||||
                    }
 | 
			
		||||
                    else
 | 
			
		||||
                    {
 | 
			
		||||
                        WriteError(client, modbusServerMessage);//返回错误码
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                else//写入
 | 
			
		||||
                {
 | 
			
		||||
                    var coreData = modbusServerMessage.Content;
 | 
			
		||||
                    if (modbusServerMessage.CurModbusAddress.ReadFunction == 1 || modbusServerMessage.CurModbusAddress.ReadFunction == 2)
 | 
			
		||||
                    {
 | 
			
		||||
                        //写入继电器
 | 
			
		||||
                        if (WriteData != null)
 | 
			
		||||
                        {
 | 
			
		||||
                            // 接收外部写入时,传出变量地址/写入字节组/转换规则/客户端
 | 
			
		||||
                            if ((await WriteData(modbusServerMessage.CurModbusAddress, modbusServerMessage.Content, ThingsGatewayBitConverter, client)).IsSuccess)
 | 
			
		||||
                            {
 | 
			
		||||
                                var result = Write(modbusServerMessage.CurModbusAddress.ToString(), coreData.ByteToBoolArray(modbusServerMessage.Length));
 | 
			
		||||
                                if (result.IsSuccess)
 | 
			
		||||
                                {
 | 
			
		||||
                                    WriteSuccess03(client, modbusServerMessage);
 | 
			
		||||
                                }
 | 
			
		||||
                                else
 | 
			
		||||
                                {
 | 
			
		||||
                                    WriteError(client, modbusServerMessage);
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                WriteError(client, modbusServerMessage);
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        else
 | 
			
		||||
                        {
 | 
			
		||||
                            //写入内存区
 | 
			
		||||
                            var result = Write(modbusServerMessage.CurModbusAddress.ToString(), coreData.ByteToBoolArray(modbusServerMessage.Length));
 | 
			
		||||
                            if (result.IsSuccess)
 | 
			
		||||
                            {
 | 
			
		||||
                                WriteSuccess03(client, modbusServerMessage);
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                WriteError(client, modbusServerMessage);
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    else
 | 
			
		||||
                    {
 | 
			
		||||
                        //写入寄存器
 | 
			
		||||
                        if (WriteData != null)
 | 
			
		||||
                        {
 | 
			
		||||
 | 
			
		||||
                            if ((await WriteData(modbusServerMessage.CurModbusAddress, modbusServerMessage.Content, ThingsGatewayBitConverter, client)).IsSuccess)
 | 
			
		||||
                            {
 | 
			
		||||
                                var result = Write(modbusServerMessage.CurModbusAddress.ToString(), coreData);
 | 
			
		||||
                                if (result.IsSuccess)
 | 
			
		||||
                                {
 | 
			
		||||
                                    WriteSuccess03(client, modbusServerMessage);
 | 
			
		||||
                                }
 | 
			
		||||
                                else
 | 
			
		||||
                                {
 | 
			
		||||
                                    WriteError(client, modbusServerMessage);
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                WriteError(client, modbusServerMessage);
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        else
 | 
			
		||||
                        {
 | 
			
		||||
                            var result = Write(modbusServerMessage.CurModbusAddress.ToString(), coreData);
 | 
			
		||||
                            if (result.IsSuccess)
 | 
			
		||||
                            {
 | 
			
		||||
                                WriteSuccess03(client, modbusServerMessage);
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                WriteError(client, modbusServerMessage);
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            Logger.LogError(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
        //返回错误码
 | 
			
		||||
        static void WriteError(SocketClient client, ModbusTcpServerMessage modbusServerMessage)
 | 
			
		||||
        {
 | 
			
		||||
            var sendData = modbusServerMessage.ReceivedBytes.SelectMiddle(0, 8)
 | 
			
		||||
.SpliceArray(new byte[] { (byte)1 });//01 lllegal function
 | 
			
		||||
            sendData[5] = (byte)(sendData.Length - 6);
 | 
			
		||||
            sendData[7] = (byte)(sendData[7] + 128);
 | 
			
		||||
            client.Send(sendData);
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private static void WriteSuccess03(SocketClient client, ModbusTcpServerMessage modbusServerMessage)
 | 
			
		||||
    {
 | 
			
		||||
        var sendData = modbusServerMessage.ReceivedBytes.SelectMiddle(0, 12);
 | 
			
		||||
        sendData[5] = (byte)(sendData.Length - 6);
 | 
			
		||||
        client.Send(sendData);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void Init(ModbusAddress mAddress)
 | 
			
		||||
    {
 | 
			
		||||
        if (ModbusServer01ByteBlocks.TryAdd(mAddress.Station, new(1024 * 128)))
 | 
			
		||||
            ModbusServer01ByteBlocks[mAddress.Station].SetLength(1024 * 128);
 | 
			
		||||
        if (ModbusServer02ByteBlocks.TryAdd(mAddress.Station, new(1024 * 128)))
 | 
			
		||||
            ModbusServer02ByteBlocks[mAddress.Station].SetLength(1024 * 128);
 | 
			
		||||
        if (ModbusServer03ByteBlocks.TryAdd(mAddress.Station, new(1024 * 128)))
 | 
			
		||||
            ModbusServer03ByteBlocks[mAddress.Station].SetLength(1024 * 128);
 | 
			
		||||
        if (ModbusServer04ByteBlocks.TryAdd(mAddress.Station, new(1024 * 128)))
 | 
			
		||||
            ModbusServer04ByteBlocks[mAddress.Station].SetLength(1024 * 128);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,4 +0,0 @@
 | 
			
		||||
<Project Sdk="Microsoft.NET.Sdk">
 | 
			
		||||
 | 
			
		||||
	
 | 
			
		||||
</Project>
 | 
			
		||||
@@ -1,129 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Extension.Generic;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Adapter.Siemens;
 | 
			
		||||
 | 
			
		||||
public partial class SiemensS7PLC : ReadWriteDevicesTcpClientBase
 | 
			
		||||
{
 | 
			
		||||
    private static OperResult<byte[]> GetWriteBitCommand(string address, bool data)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var result = SiemensAddress.ParseFrom(address);
 | 
			
		||||
            return SiemensHelper.GetWriteBitCommand(result, data);
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    private OperResult<List<byte[]>> GetReadByteCommand(string address, int length)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var from = SiemensAddress.ParseFrom(address, length);
 | 
			
		||||
            ushort num1 = 0;
 | 
			
		||||
            var listBytes = new List<byte[]>();
 | 
			
		||||
            while (num1 < length)
 | 
			
		||||
            {
 | 
			
		||||
                //pdu长度,重复生成报文,直至全部生成
 | 
			
		||||
                ushort num2 = (ushort)Math.Min(length - num1, pdu_length);
 | 
			
		||||
                from.Length = num2;
 | 
			
		||||
                var result = GetReadByteCommand(new SiemensAddress[1] { from });
 | 
			
		||||
                if (!result.IsSuccess) return new(result);
 | 
			
		||||
                listBytes.AddRange(result.Content);
 | 
			
		||||
                num1 += num2;
 | 
			
		||||
                if (from.DataCode == (byte)S7WordLength.Timer || from.DataCode == (byte)S7WordLength.Counter)
 | 
			
		||||
                {
 | 
			
		||||
                    from.Address += num2 / 2;
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    from.Address += num2 * 8;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            return OperResult.CreateSuccessResult(listBytes);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<List<byte[]>>(ex);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private OperResult<List<byte[]>> GetReadByteCommand(SiemensAddress[] siemensAddress)
 | 
			
		||||
    {
 | 
			
		||||
        if (siemensAddress.Length <= 19)
 | 
			
		||||
        {
 | 
			
		||||
            return ByteTransformUtil.GetResultFromBytes(SiemensHelper.GetReadCommand(siemensAddress), m => new List<byte[]>() { m });
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        List<byte[]> byteList = new();
 | 
			
		||||
        List<SiemensAddress[]> s7AddressDataArrayList = siemensAddress.ArraySplitByLength(19);
 | 
			
		||||
        for (int index = 0; index < s7AddressDataArrayList.Count; ++index)
 | 
			
		||||
        {
 | 
			
		||||
            var result = GetReadByteCommand(s7AddressDataArrayList[index]);
 | 
			
		||||
            if (!result.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                return result;
 | 
			
		||||
            }
 | 
			
		||||
            byteList.AddRange(result.Content);
 | 
			
		||||
        }
 | 
			
		||||
        return OperResult.CreateSuccessResult(byteList);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private OperResult<List<byte[]>> GetWriteByteCommand(string address, byte[] value)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var s_Address = SiemensAddress.ParseFrom(address);
 | 
			
		||||
 | 
			
		||||
            return GetWriteByteCommand(s_Address, value);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new(ex);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// DefalutConverter
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public static ThingsGatewayBitConverter DefalutConverter = new(BitConverter.IsLittleEndian ? EndianType.Little : EndianType.Big);
 | 
			
		||||
    private OperResult<List<byte[]>> GetWriteByteCommand(SiemensAddress address, byte[] value)
 | 
			
		||||
    {
 | 
			
		||||
        int length1 = value.Length;
 | 
			
		||||
        ushort index = 0;
 | 
			
		||||
        List<byte[]> bytes = new();
 | 
			
		||||
        while (index < length1)
 | 
			
		||||
        {
 | 
			
		||||
            //pdu长度,重复生成报文,直至全部生成
 | 
			
		||||
            ushort length2 = (ushort)Math.Min(length1 - index, pdu_length);
 | 
			
		||||
            byte[] data = DefalutConverter.ToByte(value, index, length2);
 | 
			
		||||
            OperResult<byte[]> result1 = SiemensHelper.GetWriteByteCommand(address, data);
 | 
			
		||||
            if (!result1.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                return new(result1);
 | 
			
		||||
            }
 | 
			
		||||
            bytes.Add(result1.Content);
 | 
			
		||||
            index += length2;
 | 
			
		||||
            address.Address += length2 * 8;
 | 
			
		||||
        }
 | 
			
		||||
        return OperResult.CreateSuccessResult(bytes);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,121 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Core;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// 读写扩展方法
 | 
			
		||||
/// </summary>
 | 
			
		||||
public static class ReadWriteDevicesExtensions
 | 
			
		||||
{
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 在返回的字节数组中解析每个变量的值
 | 
			
		||||
    /// 根据每个变量的<see cref="IDeviceVariableRunTime.Index"/>
 | 
			
		||||
    /// 不支持变长字符串类型变量,一定不能存在于变量List中
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="values">设备变量List</param>
 | 
			
		||||
    /// <param name="plc">设备</param>
 | 
			
		||||
    /// <param name="buffer">返回的字节数组</param>
 | 
			
		||||
    /// <param name="startIndex">开始序号</param>
 | 
			
		||||
    public static void PraseStructContent<T>(this IList<T> values, IReadWrite plc, byte[] buffer, int startIndex = 0) where T : IDeviceVariableRunTime
 | 
			
		||||
    {
 | 
			
		||||
        foreach (IDeviceVariableRunTime organizedVariable in values)
 | 
			
		||||
        {
 | 
			
		||||
            var deviceValue = organizedVariable;
 | 
			
		||||
            IThingsGatewayBitConverter byteConverter = deviceValue.ThingsGatewayBitConverter;
 | 
			
		||||
            var dataType = organizedVariable.DataTypeEnum;
 | 
			
		||||
            int index = organizedVariable.Index;
 | 
			
		||||
            switch (dataType)
 | 
			
		||||
            {
 | 
			
		||||
                case DataTypeEnum.String:
 | 
			
		||||
                    Set(organizedVariable, byteConverter.ToString(buffer, index + startIndex, byteConverter.Length ?? 1));
 | 
			
		||||
                    break;
 | 
			
		||||
                case DataTypeEnum.Boolean:
 | 
			
		||||
                    Set(organizedVariable,
 | 
			
		||||
                         byteConverter.Length > 1 ?
 | 
			
		||||
                        byteConverter.ToBoolean(buffer, index + (startIndex * 8), byteConverter.Length.Value, plc.IsBitReverse(organizedVariable.VariableAddress)) :
 | 
			
		||||
                        byteConverter.ToBoolean(buffer, index + (startIndex * 8), plc.IsBitReverse(organizedVariable.VariableAddress))
 | 
			
		||||
                        );
 | 
			
		||||
                    break;
 | 
			
		||||
                case DataTypeEnum.Byte:
 | 
			
		||||
                    Set(organizedVariable,
 | 
			
		||||
                        byteConverter.Length > 1 ?
 | 
			
		||||
                        byteConverter.ToByte(buffer, index + (startIndex), byteConverter.Length.Value) :
 | 
			
		||||
                        byteConverter.ToByte(buffer, index + startIndex));
 | 
			
		||||
                    break;
 | 
			
		||||
                case DataTypeEnum.Int16:
 | 
			
		||||
                    Set(organizedVariable,
 | 
			
		||||
                         byteConverter.Length > 1 ?
 | 
			
		||||
                        byteConverter.ToInt16(buffer, index + (startIndex), byteConverter.Length.Value) :
 | 
			
		||||
                        byteConverter.ToInt16(buffer, index + startIndex));
 | 
			
		||||
                    break;
 | 
			
		||||
                case DataTypeEnum.UInt16:
 | 
			
		||||
                    Set(organizedVariable,
 | 
			
		||||
                         byteConverter.Length > 1 ?
 | 
			
		||||
                        byteConverter.ToUInt16(buffer, index + (startIndex), byteConverter.Length.Value) :
 | 
			
		||||
                        byteConverter.ToUInt16(buffer, index + startIndex));
 | 
			
		||||
                    break;
 | 
			
		||||
                case DataTypeEnum.Int32:
 | 
			
		||||
                    Set(organizedVariable,
 | 
			
		||||
                         byteConverter.Length > 1 ?
 | 
			
		||||
                        byteConverter.ToInt32(buffer, index + (startIndex), byteConverter.Length.Value) :
 | 
			
		||||
                        byteConverter.ToInt32(buffer, index + startIndex));
 | 
			
		||||
                    break;
 | 
			
		||||
                case DataTypeEnum.UInt32:
 | 
			
		||||
                    Set(organizedVariable,
 | 
			
		||||
                         byteConverter.Length > 1 ?
 | 
			
		||||
                        byteConverter.ToUInt32(buffer, index + (startIndex), byteConverter.Length.Value) :
 | 
			
		||||
                        byteConverter.ToUInt32(buffer, index + startIndex));
 | 
			
		||||
                    break;
 | 
			
		||||
                case DataTypeEnum.Int64:
 | 
			
		||||
                    Set(organizedVariable,
 | 
			
		||||
                         byteConverter.Length > 1 ?
 | 
			
		||||
                        byteConverter.ToInt64(buffer, index + (startIndex), byteConverter.Length.Value) :
 | 
			
		||||
                        byteConverter.ToInt64(buffer, index + startIndex));
 | 
			
		||||
                    break;
 | 
			
		||||
                case DataTypeEnum.UInt64:
 | 
			
		||||
                    Set(organizedVariable,
 | 
			
		||||
                         byteConverter.Length > 1 ?
 | 
			
		||||
                        byteConverter.ToUInt64(buffer, index + (startIndex), byteConverter.Length.Value) :
 | 
			
		||||
                        byteConverter.ToUInt64(buffer, index + startIndex));
 | 
			
		||||
                    break;
 | 
			
		||||
                case DataTypeEnum.Single:
 | 
			
		||||
                    Set(organizedVariable,
 | 
			
		||||
                         byteConverter.Length > 1 ?
 | 
			
		||||
                        byteConverter.ToSingle(buffer, index + (startIndex), byteConverter.Length.Value) :
 | 
			
		||||
                        byteConverter.ToSingle(buffer, index + startIndex));
 | 
			
		||||
                    break;
 | 
			
		||||
                case DataTypeEnum.Double:
 | 
			
		||||
                    Set(organizedVariable,
 | 
			
		||||
                         byteConverter.Length > 1 ?
 | 
			
		||||
                        byteConverter.ToDouble(buffer, index + (startIndex), byteConverter.Length.Value) :
 | 
			
		||||
                        byteConverter.ToDouble(buffer, index + startIndex));
 | 
			
		||||
                    break;
 | 
			
		||||
                default:
 | 
			
		||||
                    break;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        static void Set(IDeviceVariableRunTime organizedVariable, object num)
 | 
			
		||||
        {
 | 
			
		||||
            var operResult = organizedVariable.SetValue(num); ;
 | 
			
		||||
            if (!operResult.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                throw new Exception(operResult.Message);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,159 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Core;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// TCP读写设备
 | 
			
		||||
/// </summary>
 | 
			
		||||
public abstract class ReadWriteDevicesSerialSessionBase : ReadWriteDevicesBase
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc cref="ReadWriteDevicesSerialSessionBase"/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="serialSession"></param>
 | 
			
		||||
    public ReadWriteDevicesSerialSessionBase(SerialSession serialSession)
 | 
			
		||||
    {
 | 
			
		||||
        SerialSession = serialSession;
 | 
			
		||||
        WaitingClientEx = SerialSession.GetWaitingClient(new() { ThrowBreakException = true });
 | 
			
		||||
        SerialSession.Received -= Received;
 | 
			
		||||
        SerialSession.Connecting -= Connecting;
 | 
			
		||||
        SerialSession.Connected -= Connected;
 | 
			
		||||
        SerialSession.Disconnecting -= Disconnecting;
 | 
			
		||||
        SerialSession.Disconnected -= Disconnected;
 | 
			
		||||
        SerialSession.Connecting += Connecting;
 | 
			
		||||
        SerialSession.Connected += Connected;
 | 
			
		||||
        SerialSession.Disconnecting += Disconnecting;
 | 
			
		||||
        SerialSession.Disconnected += Disconnected;
 | 
			
		||||
        SerialSession.Received += Received;
 | 
			
		||||
        Logger = SerialSession.Logger;
 | 
			
		||||
    }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 接收解析
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="client"></param>
 | 
			
		||||
    /// <param name="e"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    protected virtual Task Received(SerialSession client, ReceivedDataEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        return EasyTask.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 串口管理对象
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public SerialSession SerialSession { get; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// WaitingClientEx
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public virtual IWaitingClient<SerialSession> WaitingClientEx { get; }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void Connect(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        SerialSession.Connect();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task ConnectAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return SerialSession.ConnectAsync();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void Disconnect()
 | 
			
		||||
    {
 | 
			
		||||
        if (CascadeDisposal)
 | 
			
		||||
            SerialSession.Close();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void Dispose()
 | 
			
		||||
    {
 | 
			
		||||
        Disconnect();
 | 
			
		||||
        SerialSession.Received -= Received;
 | 
			
		||||
        SerialSession.Connecting -= Connecting;
 | 
			
		||||
        SerialSession.Connected -= Connected;
 | 
			
		||||
        SerialSession.Disconnecting -= Disconnecting;
 | 
			
		||||
        SerialSession.Disconnected -= Disconnected;
 | 
			
		||||
        if (CascadeDisposal)
 | 
			
		||||
            SerialSession.SafeDispose();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public OperResult<byte[]> SendThenResponse(byte[] data, WaitingOptions waitingOptions = null, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            waitingOptions ??= new WaitingOptions { ThrowBreakException = true };
 | 
			
		||||
            ResponsedData result = SerialSession.GetWaitingClient(waitingOptions).SendThenResponse(data, TimeOut, cancellationToken);
 | 
			
		||||
            return OperResult.CreateSuccessResult(result.Data);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<byte[]>(ex);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public async Task<OperResult<byte[]>> SendThenResponseAsync(byte[] data, WaitingOptions waitingOptions = null, CancellationToken cancellationToken = default)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            waitingOptions ??= new WaitingOptions { ThrowBreakException = true };
 | 
			
		||||
            ResponsedData result = await SerialSession.GetWaitingClient(waitingOptions).SendThenResponseAsync(data, TimeOut, cancellationToken);
 | 
			
		||||
            return OperResult.CreateSuccessResult(result.Data);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult<byte[]>(ex);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override string ToString()
 | 
			
		||||
    {
 | 
			
		||||
        return SerialSession.SerialProperty.ToString();
 | 
			
		||||
    }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Connected
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="client"></param>
 | 
			
		||||
    /// <param name="e"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    protected virtual Task Connected(ISerialSession client, ConnectedEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        Logger?.Debug(client.SerialProperty.ToString() + "连接成功");
 | 
			
		||||
        SetDataAdapter();
 | 
			
		||||
        return EasyTask.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private Task Connecting(ISerialSession client, SerialConnectingEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        Logger?.Debug(client.SerialProperty.ToString() + "正在连接");
 | 
			
		||||
        return EasyTask.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private Task Disconnected(ISerialSessionBase client, DisconnectEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        Logger?.Debug(client.SerialProperty.ToString() + "断开连接-" + e.Message);
 | 
			
		||||
        return EasyTask.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private Task Disconnecting(ISerialSessionBase client, DisconnectEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        Logger?.Debug(client.SerialProperty.ToString() + "正在主动断开连接-" + e.Message);
 | 
			
		||||
        return EasyTask.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,118 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Extension;
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// 对象拓展类
 | 
			
		||||
/// </summary>
 | 
			
		||||
public static class ObjectExtensions
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 转换布尔值
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public static bool ToBoolean(this object value, bool defaultValue = false) => value?.ToString().ToUpper() switch
 | 
			
		||||
    {
 | 
			
		||||
        "0" or "FALSE" => false,
 | 
			
		||||
        "1" or "TRUE" => true,
 | 
			
		||||
        _ => defaultValue,
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// ToLong
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public static long ToLong(this object value, long defaultValue = 0)
 | 
			
		||||
    {
 | 
			
		||||
        if (value == null || value.ToString().IsNullOrEmpty())
 | 
			
		||||
        {
 | 
			
		||||
            return defaultValue;
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            if (value is bool boolValue)
 | 
			
		||||
            {
 | 
			
		||||
                return boolValue ? 1 : 0;
 | 
			
		||||
            }
 | 
			
		||||
            return Int64.TryParse(value.ToString(), out var n) ? n : defaultValue;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// ToInt
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public static int ToInt(this object value, int defaultValue = 0)
 | 
			
		||||
    {
 | 
			
		||||
        if (value == null || value.ToString().IsNullOrEmpty())
 | 
			
		||||
        {
 | 
			
		||||
            return defaultValue;
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            if (value is bool boolValue)
 | 
			
		||||
            {
 | 
			
		||||
                return boolValue ? 1 : 0;
 | 
			
		||||
            }
 | 
			
		||||
            return int.TryParse(value.ToString(), out int n) ? n : defaultValue;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// ToDecimal
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public static decimal ToDecimal(this object value, int defaultValue = 0)
 | 
			
		||||
    {
 | 
			
		||||
        if (value is Double d)
 | 
			
		||||
        {
 | 
			
		||||
            return Double.IsNaN(d) ? defaultValue : (Decimal)d;
 | 
			
		||||
        }
 | 
			
		||||
        var str = value?.ToString();
 | 
			
		||||
        if (str.IsNullOrEmpty())
 | 
			
		||||
        {
 | 
			
		||||
            return defaultValue;
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            if (value is bool boolValue)
 | 
			
		||||
            {
 | 
			
		||||
                return boolValue ? 1 : 0;
 | 
			
		||||
            }
 | 
			
		||||
            return Decimal.TryParse(str, out var n) ? n : defaultValue;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// ToDecimal
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public static double ToDouble(this object value, double defaultValue = 0)
 | 
			
		||||
    {
 | 
			
		||||
        if (value is Double d)
 | 
			
		||||
        {
 | 
			
		||||
            return Double.IsNaN(d) ? defaultValue : (Double)d;
 | 
			
		||||
        }
 | 
			
		||||
        var str = value?.ToString();
 | 
			
		||||
        if (str.IsNullOrEmpty())
 | 
			
		||||
        {
 | 
			
		||||
            return (double)defaultValue;
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            if (value is bool boolValue)
 | 
			
		||||
            {
 | 
			
		||||
                return boolValue ? 1 : 0;
 | 
			
		||||
            }
 | 
			
		||||
            return (double)(double.TryParse(str, out var n) ? n : defaultValue);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,120 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Core
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// DateHandleAdapterExtension
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public static class DataHandlingAdapterExtension
 | 
			
		||||
    {
 | 
			
		||||
        #region SingleStreamDataHandlingAdapter
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 将<see cref="TouchSocketConfig"/>中的配置,装载在<see cref="SingleStreamDataHandlingAdapter"/>上。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="adapter"></param>
 | 
			
		||||
        /// <param name="config"></param>
 | 
			
		||||
        public static void Config(this SingleStreamDataHandlingAdapter adapter, TouchSocketConfig config)
 | 
			
		||||
        {
 | 
			
		||||
            if (config.GetValue(DataHandlingAdapterExtension.MaxPackageSizeProperty) is int v1)
 | 
			
		||||
            {
 | 
			
		||||
                adapter.MaxPackageSize = v1;
 | 
			
		||||
            }
 | 
			
		||||
            if (config.GetValue(DataHandlingAdapterExtension.CacheTimeoutProperty) != TimeSpan.Zero)
 | 
			
		||||
            {
 | 
			
		||||
                adapter.CacheTimeout = config.GetValue(DataHandlingAdapterExtension.CacheTimeoutProperty);
 | 
			
		||||
            }
 | 
			
		||||
            if (config.GetValue(DataHandlingAdapterExtension.CacheTimeoutEnableProperty) is bool v2)
 | 
			
		||||
            {
 | 
			
		||||
                adapter.CacheTimeoutEnable = v2;
 | 
			
		||||
            }
 | 
			
		||||
            if (config.GetValue(DataHandlingAdapterExtension.UpdateCacheTimeWhenRevProperty) is bool v3)
 | 
			
		||||
            {
 | 
			
		||||
                adapter.UpdateCacheTimeWhenRev = v3;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        #endregion
 | 
			
		||||
 | 
			
		||||
        #region 适配器配置
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 适配器数据包缓存启用。默认为缺省(null),如果有正常值会在设置适配器时,直接作用于<see cref="SingleStreamDataHandlingAdapter.CacheTimeout"/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public static readonly DependencyProperty<bool?> CacheTimeoutEnableProperty = DependencyProperty<bool?>.Register("CacheTimeoutEnable", null);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 适配器数据包缓存时长。默认为缺省(<see cref="TimeSpan.Zero"/>)。当该值有效时会在设置适配器时,直接作用于<see cref="SingleStreamDataHandlingAdapter.CacheTimeout"/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public static readonly DependencyProperty<TimeSpan> CacheTimeoutProperty = DependencyProperty<TimeSpan>.Register("CacheTimeout", TimeSpan.Zero);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 适配器数据包最大值。默认缺省(null),当该值有效时会在设置适配器时,直接作用于<see cref="DataHandlingAdapter.MaxPackageSize"/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public static readonly DependencyProperty<int?> MaxPackageSizeProperty = DependencyProperty<int?>.Register("MaxPackageSize", null);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 适配器数据包缓存策略。默认缺省(null),当该值有效时会在设置适配器时,直接作用于<see cref="SingleStreamDataHandlingAdapter.UpdateCacheTimeWhenRev"/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public static readonly DependencyProperty<bool?> UpdateCacheTimeWhenRevProperty = DependencyProperty<bool?>.Register("UpdateCacheTimeWhenRev", null);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 适配器数据包缓存时长。默认为缺省(<see cref="TimeSpan.Zero"/>)。当该值有效时会在设置适配器时,直接作用于<see cref="SingleStreamDataHandlingAdapter.CacheTimeout"/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="config"></param>
 | 
			
		||||
        /// <param name="value"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static TouchSocketConfig SetCacheTimeout(this TouchSocketConfig config, TimeSpan value)
 | 
			
		||||
        {
 | 
			
		||||
            config.SetValue(CacheTimeoutProperty, value);
 | 
			
		||||
            return config;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 适配器数据包缓存启用。默认为缺省(null),如果有正常值会在设置适配器时,直接作用于<see cref="SingleStreamDataHandlingAdapter.CacheTimeoutEnable"/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="config"></param>
 | 
			
		||||
        /// <param name="value"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static TouchSocketConfig SetCacheTimeoutEnable(this TouchSocketConfig config, bool value)
 | 
			
		||||
        {
 | 
			
		||||
            config.SetValue(CacheTimeoutEnableProperty, value);
 | 
			
		||||
            return config;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 适配器数据包最大值。默认缺省(null),当该值有效时会在设置适配器时,直接作用于<see cref="DataHandlingAdapter.MaxPackageSize"/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="config"></param>
 | 
			
		||||
        /// <param name="value"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static TouchSocketConfig SetMaxPackageSize(this TouchSocketConfig config, int value)
 | 
			
		||||
        {
 | 
			
		||||
            config.SetValue(MaxPackageSizeProperty, value);
 | 
			
		||||
            return config;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 适配器数据包缓存策略。默认缺省(null),当该值有效时会在设置适配器时,直接作用于<see cref="SingleStreamDataHandlingAdapter.UpdateCacheTimeWhenRev"/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="config"></param>
 | 
			
		||||
        /// <param name="value"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static TouchSocketConfig SetUpdateCacheTimeWhenRev(this TouchSocketConfig config, bool value)
 | 
			
		||||
        {
 | 
			
		||||
            config.SetValue(UpdateCacheTimeWhenRevProperty, value);
 | 
			
		||||
            return config;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        #endregion 适配器配置
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,498 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权(除特别声明或在XREF结尾的命名空间的代码)归作者本人若汝棋茗所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议,若本仓库没有设置,则按MIT开源协议授权
 | 
			
		||||
//  CSDN博客:https://blog.csdn.net/qq_40374647
 | 
			
		||||
//  哔哩哔哩视频:https://space.bilibili.com/94253567
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/RRQM_Home
 | 
			
		||||
//  Github源代码仓库:https://github.com/RRQM
 | 
			
		||||
//  API首页:http://rrqm_home.gitee.io/touchsocket/
 | 
			
		||||
//  交流QQ群:234762506
 | 
			
		||||
//  感谢您的下载和使用
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
using System.Data;
 | 
			
		||||
using System.Reflection;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Core
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// IContainerExtensions
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public static class ContainerExtension
 | 
			
		||||
    {
 | 
			
		||||
        #region RegisterSingleton
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="TFrom"></typeparam>
 | 
			
		||||
        /// <typeparam name="TTo"></typeparam>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="instance"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterSingleton<TFrom, TTo>(this IContainer container, TTo instance)
 | 
			
		||||
              where TFrom : class
 | 
			
		||||
              where TTo : class, TFrom
 | 
			
		||||
        {
 | 
			
		||||
            RegisterSingleton(container, typeof(TFrom), instance);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="instance"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterSingleton(this IContainer container, object instance)
 | 
			
		||||
        {
 | 
			
		||||
            if (instance is null)
 | 
			
		||||
            {
 | 
			
		||||
                throw new ArgumentNullException(nameof(instance));
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            RegisterSingleton(container, instance.GetType(), instance);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="fromType"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        /// <exception cref="ArgumentNullException"></exception>
 | 
			
		||||
        public static IContainer RegisterSingleton(this IContainer container, Type fromType)
 | 
			
		||||
        {
 | 
			
		||||
            if (fromType is null)
 | 
			
		||||
            {
 | 
			
		||||
                throw new ArgumentNullException(nameof(fromType));
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            RegisterSingleton(container, fromType, fromType);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="TFrom"></typeparam>
 | 
			
		||||
        /// <typeparam name="TTo"></typeparam>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <param name="instance"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterSingleton<TFrom, TTo>(this IContainer container, string key, TTo instance)
 | 
			
		||||
              where TFrom : class
 | 
			
		||||
              where TTo : class, TFrom
 | 
			
		||||
        {
 | 
			
		||||
            RegisterSingleton(container, typeof(TFrom), instance, key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="fromType"></param>
 | 
			
		||||
        /// <param name="instance"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterSingleton(this IContainer container, Type fromType, object instance, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            container.Register(new DependencyDescriptor(fromType, instance), key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="TFrom"></typeparam>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="instance"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterSingleton<TFrom>(this IContainer container, object instance, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            container.Register(new DependencyDescriptor(typeof(TFrom), instance), key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="instance"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterSingleton(this IContainer container, object instance, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            container.Register(new DependencyDescriptor(instance.GetType(), instance), key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="TFrom"></typeparam>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterSingleton<TFrom>(this IContainer container, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            container.Register(new DependencyDescriptor(typeof(TFrom), typeof(TFrom), Lifetime.Singleton), key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="fromType"></param>
 | 
			
		||||
        /// <param name="toType"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterSingleton(this IContainer container, Type fromType, Type toType, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            container.Register(new DependencyDescriptor(fromType, toType, Lifetime.Singleton), key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="func"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterSingleton<TFrom>(this IContainer container, Func<IContainer, object> func, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            container.Register(new DependencyDescriptor(typeof(TFrom), Lifetime.Singleton)
 | 
			
		||||
            {
 | 
			
		||||
                ImplementationFactory = func
 | 
			
		||||
            }, key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="fromType"></param>
 | 
			
		||||
        /// <param name="func"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterSingleton(this IContainer container, Type fromType, Func<IContainer, object> func, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            container.Register(new DependencyDescriptor(fromType, Lifetime.Singleton)
 | 
			
		||||
            {
 | 
			
		||||
                ImplementationFactory = func
 | 
			
		||||
            }, key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="TFrom"></typeparam>
 | 
			
		||||
        /// <typeparam name="TTO"></typeparam>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterSingleton<TFrom, TTO>(this IContainer container)
 | 
			
		||||
             where TFrom : class
 | 
			
		||||
             where TTO : class, TFrom
 | 
			
		||||
        {
 | 
			
		||||
            RegisterSingleton(container, typeof(TFrom), typeof(TTO));
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册单例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="TFrom"></typeparam>
 | 
			
		||||
        /// <typeparam name="TTO"></typeparam>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterSingleton<TFrom, TTO>(this IContainer container, string key)
 | 
			
		||||
             where TFrom : class
 | 
			
		||||
             where TTO : class, TFrom
 | 
			
		||||
        {
 | 
			
		||||
            RegisterSingleton(container, typeof(TFrom), typeof(TTO), key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        #endregion RegisterSingleton
 | 
			
		||||
 | 
			
		||||
        #region Transient
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册临时映射
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="TFrom"></typeparam>
 | 
			
		||||
        /// <typeparam name="TTO"></typeparam>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterTransient<TFrom, TTO>(this IContainer container)
 | 
			
		||||
             where TFrom : class
 | 
			
		||||
             where TTO : class, TFrom
 | 
			
		||||
        {
 | 
			
		||||
            RegisterTransient(container, typeof(TFrom), typeof(TTO));
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册临时映射
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="TFrom"></typeparam>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterTransient<TFrom>(this IContainer container, string key = "")
 | 
			
		||||
             where TFrom : class
 | 
			
		||||
        {
 | 
			
		||||
            RegisterTransient(container, typeof(TFrom), typeof(TFrom), key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册临时映射
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="TFrom"></typeparam>
 | 
			
		||||
        /// <typeparam name="TTO"></typeparam>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterTransient<TFrom, TTO>(this IContainer container, string key = "")
 | 
			
		||||
            where TFrom : class
 | 
			
		||||
            where TTO : class, TFrom
 | 
			
		||||
        {
 | 
			
		||||
            RegisterTransient(container, typeof(TFrom), typeof(TTO), key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册临时映射
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="fromType"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterTransient(this IContainer container, Type fromType, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            RegisterTransient(container, fromType, fromType, key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册临时映射
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="fromType"></param>
 | 
			
		||||
        /// <param name="toType"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterTransient(this IContainer container, Type fromType, Type toType, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            container.Register(new DependencyDescriptor(fromType, toType, Lifetime.Transient), key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册临时映射
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="func"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterTransient<TFrom>(this IContainer container, Func<IContainer, object> func, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            container.Register(new DependencyDescriptor(typeof(TFrom), Lifetime.Transient)
 | 
			
		||||
            {
 | 
			
		||||
                ImplementationFactory = func
 | 
			
		||||
            }, key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册临时映射
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="fromType"></param>
 | 
			
		||||
        /// <param name="func"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer RegisterTransient(this IContainer container, Type fromType, Func<IContainer, object> func, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            container.Register(new DependencyDescriptor(fromType, Lifetime.Transient)
 | 
			
		||||
            {
 | 
			
		||||
                ImplementationFactory = func
 | 
			
		||||
            }, key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        #endregion Transient
 | 
			
		||||
 | 
			
		||||
        #region Unregister
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 移除注册信息
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="fromType"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer Unregister(this IContainer container, Type fromType, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            container.Unregister(new DependencyDescriptor(fromType), key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 移除注册信息
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static IContainer Unregister<TFrom>(this IContainer container, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            container.Unregister(new DependencyDescriptor(typeof(TFrom)), key);
 | 
			
		||||
            return container;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        #endregion Unregister
 | 
			
		||||
 | 
			
		||||
        #region Resolve
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 创建类型对应的实例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="T"></typeparam>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static T Resolve<T>(this IContainer container, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            return (T)container.Resolve(typeof(T), key);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 创建<see cref="Lifetime.Transient"/>生命的未注册的根类型实例。一般适用于:目标类型没有注册,但是其成员类型已经注册的情况。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="fromType"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        /// <exception cref="Exception"></exception>
 | 
			
		||||
        public static object ResolveWithoutRoot(this IContainer container, Type fromType)
 | 
			
		||||
        {
 | 
			
		||||
            object[] ops = null;
 | 
			
		||||
            var ctor = fromType.GetConstructors().FirstOrDefault(x => x.IsDefined(typeof(DependencyInjectAttribute), true));
 | 
			
		||||
            if (ctor is null)
 | 
			
		||||
            {
 | 
			
		||||
                //如果没有被特性标记,那就取构造函数参数最多的作为注入目标
 | 
			
		||||
                if (fromType.GetConstructors().Length == 0)
 | 
			
		||||
                {
 | 
			
		||||
                    throw new Exception($"没有找到类型{fromType.FullName}的公共构造函数。");
 | 
			
		||||
                }
 | 
			
		||||
                ctor = fromType.GetConstructors().OrderByDescending(x => x.GetParameters().Length).First();
 | 
			
		||||
            }
 | 
			
		||||
            DependencyTypeAttribute dependencyTypeAttribute = null;
 | 
			
		||||
            if (fromType.IsDefined(typeof(DependencyTypeAttribute), true))
 | 
			
		||||
            {
 | 
			
		||||
                dependencyTypeAttribute = fromType.GetCustomAttribute<DependencyTypeAttribute>();
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            var parameters = ctor.GetParameters();
 | 
			
		||||
            var ps = new object[parameters.Length];
 | 
			
		||||
 | 
			
		||||
            if (dependencyTypeAttribute == null || dependencyTypeAttribute.Type.HasFlag(DependencyType.Constructor))
 | 
			
		||||
            {
 | 
			
		||||
                for (var i = 0; i < parameters.Length; i++)
 | 
			
		||||
                {
 | 
			
		||||
                    if (ops != null && ops.Length - 1 >= i)
 | 
			
		||||
                    {
 | 
			
		||||
                        ps[i] = ops[i];
 | 
			
		||||
                    }
 | 
			
		||||
                    else
 | 
			
		||||
                    {
 | 
			
		||||
                        if (parameters[i].ParameterType.IsPrimitive || parameters[i].ParameterType == typeof(string))
 | 
			
		||||
                        {
 | 
			
		||||
                            ps[i] = parameters[i].HasDefaultValue ? parameters[i].DefaultValue : default;
 | 
			
		||||
                        }
 | 
			
		||||
                        else
 | 
			
		||||
                        {
 | 
			
		||||
                            if (parameters[i].IsDefined(typeof(DependencyInjectAttribute), true))
 | 
			
		||||
                            {
 | 
			
		||||
                                var attribute = parameters[i].GetCustomAttribute<DependencyInjectAttribute>();
 | 
			
		||||
                                var type = attribute.Type ?? parameters[i].ParameterType;
 | 
			
		||||
                                ps[i] = container.Resolve(type, attribute.Key);
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                ps[i] = container.Resolve(parameters[i].ParameterType, null);
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            if (ps == null || ps.Length == 0)
 | 
			
		||||
            {
 | 
			
		||||
                return Activator.CreateInstance(fromType);
 | 
			
		||||
            }
 | 
			
		||||
            return Activator.CreateInstance(fromType, ps);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 创建<see cref="Lifetime.Transient"/>生命的未注册的根类型实例。一般适用于:目标类型没有注册,但是其成员类型已经注册的情况。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        /// <exception cref="Exception"></exception>
 | 
			
		||||
        public static T ResolveWithoutRoot<T>(this IContainer container)
 | 
			
		||||
        {
 | 
			
		||||
            return (T)ResolveWithoutRoot(container, typeof(T));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        ///  尝试创建类型对应的实例,如果类型没有注册,则会返回null或者默认值类型。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="fromType"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static object TryResolve(this IContainer container, Type fromType, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            if (container.IsRegistered(fromType))
 | 
			
		||||
            {
 | 
			
		||||
                return container.Resolve(fromType, key);
 | 
			
		||||
            }
 | 
			
		||||
            return default;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 尝试创建类型对应的实例,如果类型没有注册,则会返回null或者默认值类型。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="T"></typeparam>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static T TryResolve<T>(this IContainer container, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            return (T)TryResolve(container, typeof(T), key);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        #endregion Resolve
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,114 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.Collections;
 | 
			
		||||
using System.Collections.Concurrent;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Core
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 手动IOC容器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public abstract class ManualContainer : IContainer
 | 
			
		||||
    {
 | 
			
		||||
        private readonly ConcurrentDictionary<string, object> m_singletonInstances = new ConcurrentDictionary<string, object>();
 | 
			
		||||
 | 
			
		||||
        IEnumerator<DependencyDescriptor> IEnumerable<DependencyDescriptor>.GetEnumerator()
 | 
			
		||||
        {
 | 
			
		||||
            throw new NotImplementedException();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        IEnumerator IEnumerable.GetEnumerator()
 | 
			
		||||
        {
 | 
			
		||||
            throw new NotImplementedException();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 判断指定的类型是否已在容器中注册。
 | 
			
		||||
        /// <para>
 | 
			
		||||
        /// 在本容器中,一般均会返回<see langword="true"/>。
 | 
			
		||||
        /// </para>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="fromType"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public virtual bool IsRegistered(Type fromType, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            return true;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册描述符。
 | 
			
		||||
        /// <para>
 | 
			
		||||
        /// 一般情况下,本容器只会处理单例实例模式。
 | 
			
		||||
        /// </para>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="descriptor"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        public virtual void Register(DependencyDescriptor descriptor, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            if (descriptor.Lifetime == Lifetime.Singleton)
 | 
			
		||||
            {
 | 
			
		||||
                if (descriptor.ToInstance != null)
 | 
			
		||||
                {
 | 
			
		||||
                    this.m_singletonInstances.AddOrUpdate($"{descriptor.FromType.FullName}{key}", descriptor.ToInstance, (k, v) => descriptor.ToInstance);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        /// <exception cref="Exception"></exception>
 | 
			
		||||
        public object Resolve(Type fromType, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            if (fromType.FullName == "ThingsGateway.Foundation.Core.IContainer")
 | 
			
		||||
            {
 | 
			
		||||
                return this;
 | 
			
		||||
            }
 | 
			
		||||
            if (this.TryResolve(fromType, out var instance, key))
 | 
			
		||||
            {
 | 
			
		||||
                return instance;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            throw new Exception($"没有解决容器所需类型:{fromType.FullName}");
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 默认不实现该功能
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="descriptor"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <exception cref="NotImplementedException"></exception>
 | 
			
		||||
        public virtual void Unregister(DependencyDescriptor descriptor, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            throw new NotImplementedException();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 尝试解决Ioc容器所需类型。
 | 
			
		||||
        /// <para>
 | 
			
		||||
        /// 本方法仅实现了在单例实例注册下的获取。
 | 
			
		||||
        /// </para>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="fromType"></param>
 | 
			
		||||
        /// <param name="instance"></param>
 | 
			
		||||
        /// <param name="key"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        protected virtual bool TryResolve(Type fromType, out object instance, string key = "")
 | 
			
		||||
        {
 | 
			
		||||
            if (key.IsNullOrEmpty())
 | 
			
		||||
            {
 | 
			
		||||
                return this.m_singletonInstances.TryGetValue(fromType.FullName, out instance);
 | 
			
		||||
            }
 | 
			
		||||
            return this.m_singletonInstances.TryGetValue($"{fromType.FullName}{key}", out instance);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,66 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.Collections;
 | 
			
		||||
using System.Linq.Expressions;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Core
 | 
			
		||||
{
 | 
			
		||||
#if NET6_0_OR_GREATER
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 实例生成
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public static class InstanceCreater
 | 
			
		||||
    {
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 根据对象类型创建对象实例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="key">对象类型</param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static object Create(Type key)
 | 
			
		||||
        {
 | 
			
		||||
            return Activator.CreateInstance(key);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
#else
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 实例生成
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public static class InstanceCreater
 | 
			
		||||
    {
 | 
			
		||||
        private static readonly Hashtable m_paramCache = Hashtable.Synchronized(new Hashtable());//缓存
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 根据对象类型创建对象实例
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="key">对象类型</param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static object Create(Type key)
 | 
			
		||||
        {
 | 
			
		||||
            var value = (Func<object>)m_paramCache[key];
 | 
			
		||||
            if (value == null)
 | 
			
		||||
            {
 | 
			
		||||
                value = CreateInstanceByType(key);
 | 
			
		||||
                m_paramCache[key] = value;
 | 
			
		||||
            }
 | 
			
		||||
            return value();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        private static Func<object> CreateInstanceByType(Type type)
 | 
			
		||||
        {
 | 
			
		||||
            return Expression.Lambda<Func<object>>(Expression.New(type), null).Compile();
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
@@ -1,98 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Dmtp
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// TcpDmtpAdapter
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public class TcpDmtpAdapter : CustomFixedHeaderByteBlockDataHandlingAdapter<DmtpMessage>
 | 
			
		||||
    {
 | 
			
		||||
        private SpinLock m_locker = new SpinLock();
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override bool CanSendRequestInfo => true;
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override bool CanSplicingSend => false;
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override int HeaderLength => 6;
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        protected override DmtpMessage GetInstance()
 | 
			
		||||
        {
 | 
			
		||||
            return new DmtpMessage();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        protected override void OnReceivedSuccess(DmtpMessage request)
 | 
			
		||||
        {
 | 
			
		||||
            request.SafeDispose();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        protected override void PreviewSend(IRequestInfo requestInfo)
 | 
			
		||||
        {
 | 
			
		||||
            if (!(requestInfo is DmtpMessage message))
 | 
			
		||||
            {
 | 
			
		||||
                throw new Exception($"无法将{nameof(requestInfo)}转换为{nameof(DmtpMessage)}");
 | 
			
		||||
            }
 | 
			
		||||
            if (message.BodyByteBlock != null && message.BodyByteBlock.Length > this.MaxPackageSize)
 | 
			
		||||
            {
 | 
			
		||||
                throw new Exception("发送的BodyLength={requestInfo.BodyLength},大于设定的MaxPackageSize={this.MaxPackageSize}");
 | 
			
		||||
            }
 | 
			
		||||
            using (var byteBlock = new ByteBlock(message.GetLength()))
 | 
			
		||||
            {
 | 
			
		||||
                message.Build(byteBlock);
 | 
			
		||||
                this.GoSend(byteBlock.Buffer, 0, byteBlock.Len);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        protected override void PreviewSend(IList<ArraySegment<byte>> transferBytes)
 | 
			
		||||
        {
 | 
			
		||||
            if (transferBytes.Count == 0)
 | 
			
		||||
            {
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            var length = 0;
 | 
			
		||||
            foreach (var item in transferBytes)
 | 
			
		||||
            {
 | 
			
		||||
                length += item.Count;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (length > this.MaxPackageSize)
 | 
			
		||||
            {
 | 
			
		||||
                throw new Exception("发送数据大于设定值,相同解析器可能无法收到有效数据,已终止发送");
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            var lockTaken = false;
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                this.m_locker.Enter(ref lockTaken);
 | 
			
		||||
                foreach (var item in transferBytes)
 | 
			
		||||
                {
 | 
			
		||||
                    this.GoSend(item.Array, item.Offset, item.Count);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            finally
 | 
			
		||||
            {
 | 
			
		||||
                if (lockTaken)
 | 
			
		||||
                {
 | 
			
		||||
                    this.m_locker.Exit(false);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,42 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Dmtp
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// DmtpRouteServiceExtension
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public static class DmtpRouteServiceExtension
 | 
			
		||||
    {
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 添加Dmtp路由服务。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        public static void AddDmtpRouteService(this IContainer container)
 | 
			
		||||
        {
 | 
			
		||||
            container.RegisterSingleton<IDmtpRouteService, DmtpRouteService>();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 添加基于设定委托的Dmtp路由服务。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="func"></param>
 | 
			
		||||
        public static void AddDmtpRouteService(this IContainer container, Func<string, IDmtpActor> func)
 | 
			
		||||
        {
 | 
			
		||||
            container.RegisterSingleton<IDmtpRouteService>(new DmtpRouteService()
 | 
			
		||||
            {
 | 
			
		||||
                FindDmtpActor = func
 | 
			
		||||
            });
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,243 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权(除特别声明或在XREF结尾的命名空间的代码)归作者本人若汝棋茗所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议,若本仓库没有设置,则按MIT开源协议授权
 | 
			
		||||
//  CSDN博客:https://blog.csdn.net/qq_40374647
 | 
			
		||||
//  哔哩哔哩视频:https://space.bilibili.com/94253567
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/RRQM_Home
 | 
			
		||||
//  Github源代码仓库:https://github.com/RRQM
 | 
			
		||||
//  API首页:http://rrqm_home.gitee.io/touchsocket/
 | 
			
		||||
//  交流QQ群:234762506
 | 
			
		||||
//  感谢您的下载和使用
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
using ThingsGateway.Foundation.Resources;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Http
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Http客户端
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public class HttpClient : HttpClientBase
 | 
			
		||||
    {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Http客户端基类
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public class HttpClientBase : TcpClientBase, IHttpClient
 | 
			
		||||
    {
 | 
			
		||||
        private readonly object m_requestLocker = new object();
 | 
			
		||||
        private readonly WaitData<HttpResponse> m_waitData;
 | 
			
		||||
        private bool m_getContent;
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 构造函数
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public HttpClientBase()
 | 
			
		||||
        {
 | 
			
		||||
            this.m_waitData = new WaitData<HttpResponse>();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="timeout"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public override ITcpClient Connect(int timeout = 5000)
 | 
			
		||||
        {
 | 
			
		||||
            if (this.Config.GetValue(HttpConfigExtensions.HttpProxyProperty) is HttpProxy httpProxy)
 | 
			
		||||
            {
 | 
			
		||||
                var proxyHost = httpProxy.Host;
 | 
			
		||||
                var credential = httpProxy.Credential;
 | 
			
		||||
                var remoteHost = this.Config.GetValue(TouchSocketConfigExtension.RemoteIPHostProperty);
 | 
			
		||||
                try
 | 
			
		||||
                {
 | 
			
		||||
                    this.Config.SetRemoteIPHost(proxyHost);
 | 
			
		||||
                    base.Connect(timeout);
 | 
			
		||||
                    var request = new HttpRequest();
 | 
			
		||||
                    request.InitHeaders()
 | 
			
		||||
                        .SetHost(remoteHost.Host)
 | 
			
		||||
                        .SetUrl(remoteHost.Host, true)
 | 
			
		||||
                        .AsMethod("CONNECT");
 | 
			
		||||
                    var response = this.Request(request, timeout: timeout);
 | 
			
		||||
                    if (response.IsProxyAuthenticationRequired)
 | 
			
		||||
                    {
 | 
			
		||||
                        if (credential is null)
 | 
			
		||||
                        {
 | 
			
		||||
                            throw new Exception("未指定代理的凭据。");
 | 
			
		||||
                        }
 | 
			
		||||
                        var authHeader = response.Headers.Get(HttpHeaders.ProxyAuthenticate);
 | 
			
		||||
                        if (authHeader.IsNullOrEmpty())
 | 
			
		||||
                        {
 | 
			
		||||
                            throw new Exception("未指定代理身份验证质询。");
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                        var ares = new AuthenticationChallenge(authHeader, credential);
 | 
			
		||||
 | 
			
		||||
                        request.Headers.Add(HttpHeaders.ProxyAuthorization, ares.ToString());
 | 
			
		||||
                        if (!response.KeepAlive)
 | 
			
		||||
                        {
 | 
			
		||||
                            base.Close("代理要求关闭连接,随后重写连接。");
 | 
			
		||||
                            base.Connect(timeout);
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                        response = this.Request(request, timeout: timeout);
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    if (response.StatusCode != 200)
 | 
			
		||||
                    {
 | 
			
		||||
                        throw new Exception(response.StatusMessage);
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                finally
 | 
			
		||||
                {
 | 
			
		||||
                    this.Config.SetRemoteIPHost(remoteHost);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                base.Connect(timeout);
 | 
			
		||||
            }
 | 
			
		||||
            return this;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="request"><inheritdoc/></param>
 | 
			
		||||
        /// <param name="onlyRequest"><inheritdoc/></param>
 | 
			
		||||
        /// <param name="timeout"><inheritdoc/></param>
 | 
			
		||||
        /// <param name="token"><inheritdoc/></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public HttpResponse Request(HttpRequest request, bool onlyRequest = false, int timeout = 10 * 1000, CancellationToken token = default)
 | 
			
		||||
        {
 | 
			
		||||
            lock (this.m_requestLocker)
 | 
			
		||||
            {
 | 
			
		||||
                this.m_getContent = false;
 | 
			
		||||
                using (var byteBlock = new ByteBlock())
 | 
			
		||||
                {
 | 
			
		||||
                    request.Build(byteBlock);
 | 
			
		||||
 | 
			
		||||
                    this.m_waitData.Reset();
 | 
			
		||||
                    this.m_waitData.SetCancellationToken(token);
 | 
			
		||||
 | 
			
		||||
                    this.DefaultSend(byteBlock);
 | 
			
		||||
                    if (onlyRequest)
 | 
			
		||||
                    {
 | 
			
		||||
                        return default;
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    switch (this.m_waitData.Wait(timeout))
 | 
			
		||||
                    {
 | 
			
		||||
                        case WaitDataStatus.SetRunning:
 | 
			
		||||
                            return this.m_waitData.WaitResult;
 | 
			
		||||
 | 
			
		||||
                        case WaitDataStatus.Overtime:
 | 
			
		||||
                            throw new TimeoutException(TouchSocketHttpResource.Overtime.GetDescription());
 | 
			
		||||
                        case WaitDataStatus.Canceled:
 | 
			
		||||
                            return default;
 | 
			
		||||
 | 
			
		||||
                        case WaitDataStatus.Default:
 | 
			
		||||
                        case WaitDataStatus.Disposed:
 | 
			
		||||
                        default:
 | 
			
		||||
                            throw new Exception(TouchSocketHttpResource.UnknownError.GetDescription());
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="request"></param>
 | 
			
		||||
        /// <param name="onlyRequest"></param>
 | 
			
		||||
        /// <param name="timeout"></param>
 | 
			
		||||
        /// <param name="token"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public HttpResponse RequestContent(HttpRequest request, bool onlyRequest = false, int timeout = 10 * 1000, CancellationToken token = default)
 | 
			
		||||
        {
 | 
			
		||||
            lock (this.m_requestLocker)
 | 
			
		||||
            {
 | 
			
		||||
                this.m_getContent = true;
 | 
			
		||||
                using (var byteBlock = new ByteBlock())
 | 
			
		||||
                {
 | 
			
		||||
                    request.Build(byteBlock);
 | 
			
		||||
 | 
			
		||||
                    this.m_waitData.Reset();
 | 
			
		||||
                    this.m_waitData.SetCancellationToken(token);
 | 
			
		||||
 | 
			
		||||
                    this.DefaultSend(byteBlock);
 | 
			
		||||
                    if (onlyRequest)
 | 
			
		||||
                    {
 | 
			
		||||
                        return default;
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    switch (this.m_waitData.Wait(timeout))
 | 
			
		||||
                    {
 | 
			
		||||
                        case WaitDataStatus.SetRunning:
 | 
			
		||||
                            return this.m_waitData.WaitResult;
 | 
			
		||||
 | 
			
		||||
                        case WaitDataStatus.Overtime:
 | 
			
		||||
                            throw new TimeoutException(TouchSocketHttpResource.Overtime.GetDescription());
 | 
			
		||||
                        case WaitDataStatus.Canceled:
 | 
			
		||||
                            return default;
 | 
			
		||||
 | 
			
		||||
                        case WaitDataStatus.Default:
 | 
			
		||||
                        case WaitDataStatus.Disposed:
 | 
			
		||||
                        default:
 | 
			
		||||
                            throw new Exception(TouchSocketHttpResource.UnknownError.GetDescription());
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="disposing"></param>
 | 
			
		||||
        protected override void Dispose(bool disposing)
 | 
			
		||||
        {
 | 
			
		||||
            this.m_waitData?.Dispose();
 | 
			
		||||
            base.Dispose(disposing);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        protected override Task ReceivedData(ReceivedDataEventArgs e)
 | 
			
		||||
        {
 | 
			
		||||
            if (e.RequestInfo is HttpResponse response)
 | 
			
		||||
            {
 | 
			
		||||
                if (this.m_getContent)
 | 
			
		||||
                {
 | 
			
		||||
                    response.TryGetContent(out _);
 | 
			
		||||
                }
 | 
			
		||||
                this.m_waitData.Set(response);
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            return base.ReceivedData(e);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="e"></param>
 | 
			
		||||
        protected override async Task OnConnecting(ConnectingEventArgs e)
 | 
			
		||||
        {
 | 
			
		||||
            this.Protocol = Protocol.Http;
 | 
			
		||||
            this.SetDataHandlingAdapter(new HttpClientDataHandlingAdapter());
 | 
			
		||||
            await base.OnConnecting(e);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,148 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
#if !NET45
 | 
			
		||||
using System;
 | 
			
		||||
using System.Collections.Generic;
 | 
			
		||||
using System.Linq;
 | 
			
		||||
using System.Text;
 | 
			
		||||
using System.Threading.Tasks;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Core;
 | 
			
		||||
using ThingsGateway.Foundation.Sockets;
 | 
			
		||||
 | 
			
		||||
using HttpClient = System.Net.Http.HttpClient;
 | 
			
		||||
using HttpMethod = System.Net.Http.HttpMethod;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Http
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 这是基于<see cref="System.Net.Http.HttpClient"/>的通讯模型。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public class HttpClientSlim : DisposableObject
 | 
			
		||||
    {
 | 
			
		||||
        private readonly System.Net.Http.HttpClient m_httpClient;
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 这是基于<see cref="System.Net.Http.HttpClient"/>的通讯模型。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="httpClient"></param>
 | 
			
		||||
        public HttpClientSlim(System.Net.Http.HttpClient httpClient = default)
 | 
			
		||||
        {
 | 
			
		||||
            httpClient ??= new System.Net.Http.HttpClient();
 | 
			
		||||
            this.m_httpClient = httpClient;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 配置
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public TouchSocketConfig Config { get; private set; }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// Ioc容器
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public IContainer Container { get; private set; }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 插件管理器
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public IPluginsManager PluginsManager { get; private set; }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 日志记录器
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public ILog Logger { get; private set; }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 通讯客户端
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public System.Net.Http.HttpClient HttpClient => this.m_httpClient;
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 加载配置
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="config"></param>
 | 
			
		||||
        protected virtual void LoadConfig(TouchSocketConfig config)
 | 
			
		||||
        {
 | 
			
		||||
            this.m_httpClient.BaseAddress ??= config.GetValue(TouchSocketConfigExtension.RemoteIPHostProperty);
 | 
			
		||||
            this.Logger ??= this.Container.Resolve<ILog>();
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 配置
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="config"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        /// <exception cref="ArgumentNullException"></exception>
 | 
			
		||||
        public HttpClientSlim Setup(TouchSocketConfig config)
 | 
			
		||||
        {
 | 
			
		||||
            if (config == null)
 | 
			
		||||
            {
 | 
			
		||||
                throw new ArgumentNullException(nameof(config));
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            this.ThrowIfDisposed();
 | 
			
		||||
 | 
			
		||||
            this.BuildConfig(config);
 | 
			
		||||
 | 
			
		||||
            this.PluginsManager.Raise(nameof(ILoadingConfigPlugin.OnLoadingConfig), this, new ConfigEventArgs(config));
 | 
			
		||||
            this.LoadConfig(this.Config);
 | 
			
		||||
            this.PluginsManager.Raise(nameof(ILoadedConfigPlugin.OnLoadedConfig), this, new ConfigEventArgs(config));
 | 
			
		||||
 | 
			
		||||
            return this;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        private void BuildConfig(TouchSocketConfig config)
 | 
			
		||||
        {
 | 
			
		||||
            this.Config = config;
 | 
			
		||||
 | 
			
		||||
            if (!(config.GetValue(TouchSocketCoreConfigExtension.ContainerProperty) is IContainer container))
 | 
			
		||||
            {
 | 
			
		||||
                container = new Container();
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (!container.IsRegistered(typeof(ILog)))
 | 
			
		||||
            {
 | 
			
		||||
                container.RegisterSingleton<ILog, LoggerGroup>();
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (!(config.GetValue(TouchSocketCoreConfigExtension.PluginsManagerProperty) is IPluginsManager pluginsManager))
 | 
			
		||||
            {
 | 
			
		||||
                pluginsManager = new PluginsManager(container);
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (container.IsRegistered(typeof(IPluginsManager)))
 | 
			
		||||
            {
 | 
			
		||||
                pluginsManager = container.Resolve<IPluginsManager>();
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                container.RegisterSingleton<IPluginsManager>(pluginsManager);
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (config.GetValue(TouchSocketCoreConfigExtension.ConfigureContainerProperty) is Action<IContainer> actionContainer)
 | 
			
		||||
            {
 | 
			
		||||
                actionContainer.Invoke(container);
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (config.GetValue(TouchSocketCoreConfigExtension.ConfigurePluginsProperty) is Action<IPluginsManager> actionPluginsManager)
 | 
			
		||||
            {
 | 
			
		||||
                pluginsManager.Enable = true;
 | 
			
		||||
                actionPluginsManager.Invoke(pluginsManager);
 | 
			
		||||
            }
 | 
			
		||||
            this.Container = container;
 | 
			
		||||
            this.PluginsManager = pluginsManager;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
@@ -1,60 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权(除特别声明或在XREF结尾的命名空间的代码)归作者本人若汝棋茗所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议,若本仓库没有设置,则按MIT开源协议授权
 | 
			
		||||
//  CSDN博客:https://blog.csdn.net/qq_40374647
 | 
			
		||||
//  哔哩哔哩视频:https://space.bilibili.com/94253567
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/RRQM_Home
 | 
			
		||||
//  Github源代码仓库:https://github.com/RRQM
 | 
			
		||||
//  API首页:http://rrqm_home.gitee.io/touchsocket/
 | 
			
		||||
//  交流QQ群:234762506
 | 
			
		||||
//  感谢您的下载和使用
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Rpc
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// RpcServerFactory
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public class RpcServerFactory : IRpcServerFactory
 | 
			
		||||
    {
 | 
			
		||||
        private readonly IContainer m_container;
 | 
			
		||||
        private readonly ILog m_logger;
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 构造函数
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        /// <param name="logger"></param>
 | 
			
		||||
        public RpcServerFactory(IContainer container, ILog logger)
 | 
			
		||||
        {
 | 
			
		||||
            this.m_container = container;
 | 
			
		||||
            this.m_logger = logger;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        IRpcServer IRpcServerFactory.Create(ICallContext callContext, object[] ps)
 | 
			
		||||
        {
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                return (IRpcServer)this.m_container.Resolve(callContext.MethodInstance.ServerFromType);
 | 
			
		||||
            }
 | 
			
		||||
            catch (System.Exception ex)
 | 
			
		||||
            {
 | 
			
		||||
                this.m_logger.Exception(ex);
 | 
			
		||||
                throw;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,536 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权(除特别声明或在XREF结尾的命名空间的代码)归作者本人若汝棋茗所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议,若本仓库没有设置,则按MIT开源协议授权
 | 
			
		||||
//  CSDN博客:https://blog.csdn.net/qq_40374647
 | 
			
		||||
//  哔哩哔哩视频:https://space.bilibili.com/94253567
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/RRQM_Home
 | 
			
		||||
//  Github源代码仓库:https://github.com/RRQM
 | 
			
		||||
//  API首页:http://rrqm_home.gitee.io/touchsocket/
 | 
			
		||||
//  交流QQ群:234762506
 | 
			
		||||
//  感谢您的下载和使用
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
using System.Collections;
 | 
			
		||||
using System.Collections.Concurrent;
 | 
			
		||||
using System.Reflection;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Rpc
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Rpc仓库
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public class RpcStore : DisposableObject, IEnumerable<IRpcParser>
 | 
			
		||||
    {
 | 
			
		||||
        private readonly ConcurrentList<IRpcParser> m_parsers = new ConcurrentList<IRpcParser>();
 | 
			
		||||
        private readonly ConcurrentDictionary<Type, List<MethodInstance>> m_serverTypes = new ConcurrentDictionary<Type, List<MethodInstance>>();
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 实例化一个Rpc仓库。
 | 
			
		||||
        /// <para>需要指定<see cref="IContainer"/>容器。一般和对应的服务器、客户端共用一个容器比较好。</para>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public RpcStore(IContainer container)
 | 
			
		||||
        {
 | 
			
		||||
            this.Container = container ?? throw new ArgumentNullException(nameof(container));
 | 
			
		||||
            if (!container.IsRegistered(typeof(IRpcServerFactory)))
 | 
			
		||||
            {
 | 
			
		||||
                this.Container.RegisterSingleton<IRpcServerFactory, RpcServerFactory>();
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 内置IOC容器
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public IContainer Container { get; private set; }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 服务类型
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public Type[] ServerTypes => this.m_serverTypes.Keys.ToArray();
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 执行Rpc
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="rpcServer"></param>
 | 
			
		||||
        /// <param name="ps"></param>
 | 
			
		||||
        /// <param name="callContext"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static InvokeResult Execute(IRpcServer rpcServer, object[] ps, ICallContext callContext)
 | 
			
		||||
        {
 | 
			
		||||
            var invokeResult = new InvokeResult();
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                if (callContext.MethodInstance.Filters != null)
 | 
			
		||||
                {
 | 
			
		||||
                    for (var i = 0; i < callContext.MethodInstance.Filters.Length; i++)
 | 
			
		||||
                    {
 | 
			
		||||
                        invokeResult = callContext.MethodInstance.Filters[i].ExecutingAsync(callContext, ps, invokeResult)
 | 
			
		||||
                            .ConfigureAwait(false).GetAwaiter().GetResult();
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                if (invokeResult.Status != InvokeStatus.Ready)
 | 
			
		||||
                {
 | 
			
		||||
                    return invokeResult;
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                //调用
 | 
			
		||||
                switch (callContext.MethodInstance.TaskType)
 | 
			
		||||
                {
 | 
			
		||||
                    case TaskReturnType.Task:
 | 
			
		||||
                        {
 | 
			
		||||
                            callContext.MethodInstance.InvokeAsync(rpcServer, ps)
 | 
			
		||||
                                .ConfigureAwait(false).GetAwaiter().GetResult();
 | 
			
		||||
                        }
 | 
			
		||||
                        break;
 | 
			
		||||
 | 
			
		||||
                    case TaskReturnType.TaskObject:
 | 
			
		||||
                        {
 | 
			
		||||
                            invokeResult.Result = callContext.MethodInstance.InvokeObjectAsync(rpcServer, ps)
 | 
			
		||||
                                 .ConfigureAwait(false).GetAwaiter().GetResult();
 | 
			
		||||
                        }
 | 
			
		||||
                        break;
 | 
			
		||||
 | 
			
		||||
                    default:
 | 
			
		||||
                    case TaskReturnType.None:
 | 
			
		||||
                        {
 | 
			
		||||
                            if (callContext.MethodInstance.HasReturn)
 | 
			
		||||
                            {
 | 
			
		||||
                                invokeResult.Result = callContext.MethodInstance.Invoke(rpcServer, ps);
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                callContext.MethodInstance.Invoke(rpcServer, ps);
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        break;
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                invokeResult.Status = InvokeStatus.Success;
 | 
			
		||||
                if (callContext.MethodInstance.Filters != null)
 | 
			
		||||
                {
 | 
			
		||||
                    for (var i = 0; i < callContext.MethodInstance.Filters.Length; i++)
 | 
			
		||||
                    {
 | 
			
		||||
                        invokeResult = callContext.MethodInstance.Filters[i].ExecutedAsync(callContext, ps, invokeResult)
 | 
			
		||||
                            .ConfigureAwait(false).GetAwaiter().GetResult();
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            catch (TargetInvocationException ex)
 | 
			
		||||
            {
 | 
			
		||||
                invokeResult.Status = InvokeStatus.InvocationException;
 | 
			
		||||
                invokeResult.Message = ex.InnerException != null ? "函数内部发生异常,信息:" + ex.InnerException.Message : "函数内部发生异常,信息:未知";
 | 
			
		||||
                if (callContext.MethodInstance.Filters != null)
 | 
			
		||||
                {
 | 
			
		||||
                    for (var i = 0; i < callContext.MethodInstance.Filters.Length; i++)
 | 
			
		||||
                    {
 | 
			
		||||
                        invokeResult = callContext.MethodInstance.Filters[i].ExecutExceptionAsync(callContext, ps, invokeResult, ex)
 | 
			
		||||
                             .ConfigureAwait(false).GetAwaiter().GetResult();
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            catch (Exception ex)
 | 
			
		||||
            {
 | 
			
		||||
                invokeResult.Status = InvokeStatus.Exception;
 | 
			
		||||
                invokeResult.Message = ex.Message;
 | 
			
		||||
                if (callContext.MethodInstance.Filters != null)
 | 
			
		||||
                {
 | 
			
		||||
                    for (var i = 0; i < callContext.MethodInstance.Filters.Length; i++)
 | 
			
		||||
                    {
 | 
			
		||||
                        invokeResult = callContext.MethodInstance.Filters[i].ExecutExceptionAsync(callContext, ps, invokeResult, ex).ConfigureAwait(false).GetAwaiter().GetResult();
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            return invokeResult;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 异步执行Rpc
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="rpcServer"></param>
 | 
			
		||||
        /// <param name="ps"></param>
 | 
			
		||||
        /// <param name="callContext"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static async Task<InvokeResult> ExecuteAsync(IRpcServer rpcServer, object[] ps, ICallContext callContext)
 | 
			
		||||
        {
 | 
			
		||||
            var invokeResult = new InvokeResult();
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                if (callContext.MethodInstance.Filters != null)
 | 
			
		||||
                {
 | 
			
		||||
                    for (var i = 0; i < callContext.MethodInstance.Filters.Length; i++)
 | 
			
		||||
                    {
 | 
			
		||||
                        invokeResult = await callContext.MethodInstance.Filters[i].ExecutingAsync(callContext, ps, invokeResult);
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                if (invokeResult.Status != InvokeStatus.Ready)
 | 
			
		||||
                {
 | 
			
		||||
                    return invokeResult;
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                //调用
 | 
			
		||||
                switch (callContext.MethodInstance.TaskType)
 | 
			
		||||
                {
 | 
			
		||||
                    case TaskReturnType.Task:
 | 
			
		||||
                        {
 | 
			
		||||
                            await (Task)callContext.MethodInstance.Invoke(rpcServer, ps);
 | 
			
		||||
                        }
 | 
			
		||||
                        break;
 | 
			
		||||
 | 
			
		||||
                    case TaskReturnType.TaskObject:
 | 
			
		||||
                        {
 | 
			
		||||
                            invokeResult.Result = await callContext.MethodInstance.InvokeObjectAsync(rpcServer, ps);
 | 
			
		||||
                        }
 | 
			
		||||
                        break;
 | 
			
		||||
 | 
			
		||||
                    default:
 | 
			
		||||
                    case TaskReturnType.None:
 | 
			
		||||
                        {
 | 
			
		||||
                            if (callContext.MethodInstance.HasReturn)
 | 
			
		||||
                            {
 | 
			
		||||
                                invokeResult.Result = callContext.MethodInstance.Invoke(rpcServer, ps);
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                callContext.MethodInstance.Invoke(rpcServer, ps);
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        break;
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                invokeResult.Status = InvokeStatus.Success;
 | 
			
		||||
                if (callContext.MethodInstance.Filters != null)
 | 
			
		||||
                {
 | 
			
		||||
                    for (var i = 0; i < callContext.MethodInstance.Filters.Length; i++)
 | 
			
		||||
                    {
 | 
			
		||||
                        invokeResult = await callContext.MethodInstance.Filters[i].ExecutedAsync(callContext, ps, invokeResult);
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            catch (TargetInvocationException ex)
 | 
			
		||||
            {
 | 
			
		||||
                invokeResult.Status = InvokeStatus.InvocationException;
 | 
			
		||||
                invokeResult.Message = ex.InnerException != null ? "函数内部发生异常,信息:" + ex.InnerException.Message : "函数内部发生异常,信息:未知";
 | 
			
		||||
                if (callContext.MethodInstance.Filters != null)
 | 
			
		||||
                {
 | 
			
		||||
                    for (var i = 0; i < callContext.MethodInstance.Filters.Length; i++)
 | 
			
		||||
                    {
 | 
			
		||||
                        invokeResult = await callContext.MethodInstance.Filters[i].ExecutExceptionAsync(callContext, ps, invokeResult, ex);
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            catch (Exception ex)
 | 
			
		||||
            {
 | 
			
		||||
                invokeResult.Status = InvokeStatus.Exception;
 | 
			
		||||
                invokeResult.Message = ex.Message;
 | 
			
		||||
                if (callContext.MethodInstance.Filters != null)
 | 
			
		||||
                {
 | 
			
		||||
                    for (var i = 0; i < callContext.MethodInstance.Filters.Length; i++)
 | 
			
		||||
                    {
 | 
			
		||||
                        invokeResult = await callContext.MethodInstance.Filters[i].ExecutExceptionAsync(callContext, ps, invokeResult, ex);
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            return invokeResult;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 添加Rpc解析器
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="parser">解析器实例</param>
 | 
			
		||||
        /// <param name="applyServer">是否应用已注册服务</param>
 | 
			
		||||
        public void AddRpcParser(IRpcParser parser, bool applyServer = true)
 | 
			
		||||
        {
 | 
			
		||||
            this.ThrowIfDisposed();
 | 
			
		||||
            this.m_parsers.Add(parser);
 | 
			
		||||
            //parser.SetRpcStore(this);
 | 
			
		||||
            if (applyServer)
 | 
			
		||||
            {
 | 
			
		||||
                foreach (var item in this.m_serverTypes)
 | 
			
		||||
                {
 | 
			
		||||
                    parser.OnRegisterServer(item.Value.ToArray());
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 获取所有已注册的函数。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public MethodInstance[] GetAllMethods()
 | 
			
		||||
        {
 | 
			
		||||
            var methods = new List<MethodInstance>();
 | 
			
		||||
            foreach (var item in this.m_serverTypes.Values)
 | 
			
		||||
            {
 | 
			
		||||
                methods.AddRange(item);
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            return methods.ToArray();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        IEnumerator IEnumerable.GetEnumerator()
 | 
			
		||||
        {
 | 
			
		||||
            return this.m_parsers.GetEnumerator();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 返回枚举对象
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        IEnumerator<IRpcParser> IEnumerable<IRpcParser>.GetEnumerator()
 | 
			
		||||
        {
 | 
			
		||||
            return this.m_parsers.GetEnumerator();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 本地获取代理
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="namespace"></param>
 | 
			
		||||
        /// <param name="attrbuteTypes"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public string GetProxyCodes(string @namespace, params Type[] attrbuteTypes)
 | 
			
		||||
        {
 | 
			
		||||
            var cellCodes = this.GetProxyInfo(attrbuteTypes);
 | 
			
		||||
            return CodeGenerator.ConvertToCode(@namespace, cellCodes);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 获取生成的代理
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="TAttribute"></typeparam>
 | 
			
		||||
        /// <param name="namespace"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public string GetProxyCodes<TAttribute>(string @namespace) where TAttribute : RpcAttribute
 | 
			
		||||
        {
 | 
			
		||||
            var cellCodes = this.GetProxyInfo(new Type[] { typeof(TAttribute) });
 | 
			
		||||
            return CodeGenerator.ConvertToCode(@namespace, cellCodes);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 从本地获取代理
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="attrbuteType"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public ServerCellCode[] GetProxyInfo(Type[] attrbuteType)
 | 
			
		||||
        {
 | 
			
		||||
            if (this.DisposedValue)
 | 
			
		||||
            {
 | 
			
		||||
                throw new ObjectDisposedException(this.GetType().FullName);
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            var codes = new List<ServerCellCode>();
 | 
			
		||||
 | 
			
		||||
            foreach (var attrbute in attrbuteType)
 | 
			
		||||
            {
 | 
			
		||||
                foreach (var item in this.m_serverTypes.Keys)
 | 
			
		||||
                {
 | 
			
		||||
                    var serverCellCode = CodeGenerator.Generator(item, attrbute);
 | 
			
		||||
                    codes.Add(serverCellCode);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            return codes.ToArray();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 获取服务类型对应的服务方法。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="serverType"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public MethodInstance[] GetServerMethodInstances(Type serverType)
 | 
			
		||||
        {
 | 
			
		||||
            return this.m_serverTypes[serverType].ToArray();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 移除Rpc解析器
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="parser"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public bool RemoveRpcParser(IRpcParser parser)
 | 
			
		||||
        {
 | 
			
		||||
            return this.m_parsers.Remove(parser);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 移除注册服务
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="provider"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public int UnregisterServer(IRpcServer provider)
 | 
			
		||||
        {
 | 
			
		||||
            return this.UnregisterServer(provider.GetType());
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 移除注册服务
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="providerType"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public int UnregisterServer(Type providerType)
 | 
			
		||||
        {
 | 
			
		||||
            this.ThrowIfDisposed();
 | 
			
		||||
            if (!typeof(IRpcServer).IsAssignableFrom(providerType))
 | 
			
		||||
            {
 | 
			
		||||
                throw new RpcException("类型不相符");
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (this.RemoveServer(providerType, out var instances))
 | 
			
		||||
            {
 | 
			
		||||
                foreach (var parser in this)
 | 
			
		||||
                {
 | 
			
		||||
                    parser.OnUnregisterServer(instances);
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                return instances.Length;
 | 
			
		||||
            }
 | 
			
		||||
            return 0;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 移除注册服务
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <typeparam name="T"></typeparam>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public int UnregisterServer<T>() where T : IRpcServer
 | 
			
		||||
        {
 | 
			
		||||
            return this.UnregisterServer(typeof(T));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="disposing"></param>
 | 
			
		||||
        protected override void Dispose(bool disposing)
 | 
			
		||||
        {
 | 
			
		||||
            if (!this.DisposedValue)
 | 
			
		||||
            {
 | 
			
		||||
                foreach (var item in this)
 | 
			
		||||
                {
 | 
			
		||||
                    item.SafeDispose();
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            base.Dispose(disposing);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        private bool RemoveServer(Type type, out MethodInstance[] methodInstances)
 | 
			
		||||
        {
 | 
			
		||||
            foreach (var newType in this.m_serverTypes.Keys)
 | 
			
		||||
            {
 | 
			
		||||
                if (newType.FullName == type.FullName)
 | 
			
		||||
                {
 | 
			
		||||
                    this.m_serverTypes.TryRemove(newType, out var list);
 | 
			
		||||
                    methodInstances = list.ToArray();
 | 
			
		||||
                    return true;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            methodInstances = null;
 | 
			
		||||
            return false;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        #region 注册
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册为单例服务
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="serverFromType"></param>
 | 
			
		||||
        /// <param name="rpcServer"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public void RegisterServer(Type serverFromType, IRpcServer rpcServer)
 | 
			
		||||
        {
 | 
			
		||||
            if (!typeof(IRpcServer).IsAssignableFrom(serverFromType))
 | 
			
		||||
            {
 | 
			
		||||
                throw new RpcException($"注册类型必须与{nameof(IRpcServer)}有继承关系");
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (!serverFromType.IsAssignableFrom(rpcServer.GetType()))
 | 
			
		||||
            {
 | 
			
		||||
                throw new RpcException("实例类型必须与注册类型有继承关系。");
 | 
			
		||||
            }
 | 
			
		||||
            foreach (var item in this.m_serverTypes.Keys)
 | 
			
		||||
            {
 | 
			
		||||
                if (item.FullName == serverFromType.FullName)
 | 
			
		||||
                {
 | 
			
		||||
                    return;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            var methodInstances = CodeGenerator.GetMethodInstances(serverFromType, rpcServer.GetType());
 | 
			
		||||
            foreach (var item in methodInstances)
 | 
			
		||||
            {
 | 
			
		||||
                item.ServerFactory = this.Container.Resolve<IRpcServerFactory>() ?? throw new ArgumentNullException($"{nameof(IRpcServerFactory)}");
 | 
			
		||||
            }
 | 
			
		||||
            this.m_serverTypes.TryAdd(serverFromType, new List<MethodInstance>(methodInstances));
 | 
			
		||||
            this.Container.RegisterSingleton(serverFromType, rpcServer);
 | 
			
		||||
 | 
			
		||||
            foreach (var parser in this)
 | 
			
		||||
            {
 | 
			
		||||
                parser.OnRegisterServer(methodInstances);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 注册服务
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="serverFromType"></param>
 | 
			
		||||
        /// <param name="serverToType"></param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public void RegisterServer(Type serverFromType, Type serverToType)
 | 
			
		||||
        {
 | 
			
		||||
            if (!typeof(IRpcServer).IsAssignableFrom(serverFromType))
 | 
			
		||||
            {
 | 
			
		||||
                throw new RpcException($"注册类型必须与{nameof(IRpcServer)}有继承关系");
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (!serverFromType.IsAssignableFrom(serverToType))
 | 
			
		||||
            {
 | 
			
		||||
                throw new RpcException("实例类型必须与注册类型有继承关系。");
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            foreach (var item in this.m_serverTypes.Keys)
 | 
			
		||||
            {
 | 
			
		||||
                if (item.FullName == serverFromType.FullName)
 | 
			
		||||
                {
 | 
			
		||||
                    return;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
            if (typeof(ITransientRpcServer).IsAssignableFrom(serverFromType))
 | 
			
		||||
            {
 | 
			
		||||
                this.Container.RegisterTransient(serverFromType, serverToType);
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                this.Container.RegisterSingleton(serverFromType, serverToType);
 | 
			
		||||
            }
 | 
			
		||||
            var methodInstances = CodeGenerator.GetMethodInstances(serverFromType, serverToType);
 | 
			
		||||
            foreach (var item in methodInstances)
 | 
			
		||||
            {
 | 
			
		||||
                item.ServerFactory = this.Container.Resolve<IRpcServerFactory>() ?? throw new ArgumentNullException($"{nameof(IRpcServerFactory)}");
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            this.m_serverTypes.TryAdd(serverFromType, new List<MethodInstance>(methodInstances));
 | 
			
		||||
 | 
			
		||||
            foreach (var parser in this)
 | 
			
		||||
            {
 | 
			
		||||
                parser.OnRegisterServer(methodInstances);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        #endregion 注册
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,93 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权(除特别声明或在XREF结尾的命名空间的代码)归作者本人若汝棋茗所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议,若本仓库没有设置,则按MIT开源协议授权
 | 
			
		||||
//  CSDN博客:https://blog.csdn.net/qq_40374647
 | 
			
		||||
//  哔哩哔哩视频:https://space.bilibili.com/94253567
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/RRQM_Home
 | 
			
		||||
//  Github源代码仓库:https://github.com/RRQM
 | 
			
		||||
//  API首页:http://rrqm_home.gitee.io/touchsocket/
 | 
			
		||||
//  交流QQ群:234762506
 | 
			
		||||
//  感谢您的下载和使用
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Rpc
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Rpc异常
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Serializable]
 | 
			
		||||
    public class RpcException : Exception
 | 
			
		||||
    {
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        ///构造函数
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public RpcException() : base() { }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        ///构造函数
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="message"></param>
 | 
			
		||||
        public RpcException(string message) : base(message) { }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        ///构造函数
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="message"></param>
 | 
			
		||||
        /// <param name="inner"></param>
 | 
			
		||||
        public RpcException(string message, System.Exception inner) : base(message, inner) { }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        ///构造函数
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="info"></param>
 | 
			
		||||
        /// <param name="context"></param>
 | 
			
		||||
        protected RpcException(System.Runtime.Serialization.SerializationInfo info,
 | 
			
		||||
            System.Runtime.Serialization.StreamingContext context) : base(info, context) { }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Rpc调用异常
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Serializable]
 | 
			
		||||
    public class RpcInvokeException : Exception
 | 
			
		||||
    {
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        ///构造函数
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public RpcInvokeException() : base() { }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        ///构造函数
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="message"></param>
 | 
			
		||||
        public RpcInvokeException(string message) : base(message) { }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        ///构造函数
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="message"></param>
 | 
			
		||||
        /// <param name="inner"></param>
 | 
			
		||||
        public RpcInvokeException(string message, System.Exception inner) : base(message, inner) { }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        ///构造函数
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="info"></param>
 | 
			
		||||
        /// <param name="context"></param>
 | 
			
		||||
        protected RpcInvokeException(System.Runtime.Serialization.SerializationInfo info,
 | 
			
		||||
            System.Runtime.Serialization.StreamingContext context) : base(info, context) { }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,48 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Rpc
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 全局Rpc仓库配置插件。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public class GlobalRpcStorePlugin : PluginBase
 | 
			
		||||
    {
 | 
			
		||||
        private readonly RpcStore m_rpcStore;
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 全局Rpc仓库配置插件。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="container"></param>
 | 
			
		||||
        public GlobalRpcStorePlugin(IContainer container)
 | 
			
		||||
        {
 | 
			
		||||
            if (container.IsRegistered(typeof(RpcStore)))
 | 
			
		||||
            {
 | 
			
		||||
                this.m_rpcStore = container.Resolve<RpcStore>();
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                this.m_rpcStore = new RpcStore(container);
 | 
			
		||||
                container.RegisterSingleton<RpcStore>(this.m_rpcStore);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 全局配置Rpc服务
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="action"></param>
 | 
			
		||||
        public void ConfigureRpcStore(Action<RpcStore> action)
 | 
			
		||||
        {
 | 
			
		||||
            action?.Invoke(this.m_rpcStore);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,48 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.IO.Ports;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Serial;
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// Connecting
 | 
			
		||||
/// </summary>
 | 
			
		||||
/// <typeparam name="TClient"></typeparam>
 | 
			
		||||
/// <param name="client"></param>
 | 
			
		||||
/// <param name="e"></param>
 | 
			
		||||
public delegate Task SerialConnectingEventHandler<TClient>(TClient client, SerialConnectingEventArgs e);
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// 客户端连接事件。
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class SerialConnectingEventArgs : MsgPermitEventArgs
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 构造函数
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="serialPort"></param>
 | 
			
		||||
    public SerialConnectingEventArgs(SerialPort serialPort)
 | 
			
		||||
    {
 | 
			
		||||
        this.SerialPort = serialPort;
 | 
			
		||||
        this.IsPermitOperation = true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 客户端Id。该Id的赋值,仅在服务器适用。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public string Id { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 新初始化的通信器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public SerialPort SerialPort { get; private set; }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,38 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// 串口附加属性
 | 
			
		||||
/// </summary>
 | 
			
		||||
public static class SerialConfigExtension
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 串口属性
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public static readonly DependencyProperty<SerialProperty> SerialProperty =
 | 
			
		||||
        DependencyProperty<SerialProperty>.Register("SerialProperty", new());
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 设置串口
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="config"></param>
 | 
			
		||||
    /// <param name="value"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public static TouchSocketConfig SetSerialProperty(this TouchSocketConfig config, SerialProperty value)
 | 
			
		||||
    {
 | 
			
		||||
        config.SetValue(SerialProperty, value);
 | 
			
		||||
        return config;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,58 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.IO.Ports;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// SocketExtension
 | 
			
		||||
/// </summary>
 | 
			
		||||
public static class SerialPortExtensions
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 会使用同步锁,保证所有数据上缓存区。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="serialPort"></param>
 | 
			
		||||
    /// <param name="buffer"></param>
 | 
			
		||||
    /// <param name="offset"></param>
 | 
			
		||||
    /// <param name="length"></param>
 | 
			
		||||
    public static void AbsoluteSend(this SerialPort serialPort, byte[] buffer, int offset, int length)
 | 
			
		||||
    {
 | 
			
		||||
        lock (serialPort)
 | 
			
		||||
        {
 | 
			
		||||
            serialPort.Write(buffer, offset, length);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 尝试关闭<see cref="SerialPort"/>。不会抛出异常。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="serialPort"></param>
 | 
			
		||||
    public static void TryClose(this SerialPort serialPort)
 | 
			
		||||
    {
 | 
			
		||||
        lock (serialPort)
 | 
			
		||||
        {
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                if (serialPort.IsOpen)
 | 
			
		||||
                {
 | 
			
		||||
                    serialPort.Close();
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            catch
 | 
			
		||||
            {
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,43 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// <inheritdoc cref="ISerialSessionBase"/>
 | 
			
		||||
/// </summary>
 | 
			
		||||
public interface ISerialSession : ISerialSessionBase, IClientSender, IPluginObject
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 成功打开串口
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    ConnectedEventHandler<ISerialSession> Connected { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 准备连接串口的时候
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    SerialConnectingEventHandler<ISerialSession> Connecting { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 连接串口
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <exception cref="Exception"></exception>
 | 
			
		||||
    ISerialSession Connect();
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 配置服务器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="config"></param>
 | 
			
		||||
    /// <exception cref="Exception"></exception>
 | 
			
		||||
    ISerialSession Setup(TouchSocketConfig config);
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,76 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.IO.Ports;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// 串口连接接口。
 | 
			
		||||
/// </summary>
 | 
			
		||||
public interface ISerialSessionBase : IClient, ISender, IDefaultSender, IPluginObject, IRequsetInfoSender
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 是否允许自由调用<see cref="SetDataHandlingAdapter"/>进行赋值。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    bool CanSetDataHandlingAdapter { get; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 客户端配置
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    TouchSocketConfig Config { get; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 数据处理适配器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    SingleStreamDataHandlingAdapter DataHandlingAdapter { get; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 断开连接
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    DisconnectEventHandler<ISerialSessionBase> Disconnected { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 即将断开连接(仅主动断开时有效)。
 | 
			
		||||
    /// <para>
 | 
			
		||||
    /// </para>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    DisconnectEventHandler<ISerialSessionBase> Disconnecting { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 主通信器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    SerialPort MainSerialPort { get; }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 判断是否在线
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    bool Online { get; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 串口描述
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    SerialProperty SerialProperty { get; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 关闭客户端。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="msg"></param>
 | 
			
		||||
    /// <exception cref="Exception"></exception>
 | 
			
		||||
    void Close(string msg = TouchSocketCoreUtility.Empty);
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 设置数据处理适配器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="adapter"></param>
 | 
			
		||||
    void SetDataHandlingAdapter(SingleStreamDataHandlingAdapter adapter);
 | 
			
		||||
}
 | 
			
		||||
@@ -1,43 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// 通讯基类
 | 
			
		||||
/// </summary>
 | 
			
		||||
public abstract class BaseSerial : DependencyObject, ISerial
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 同步根。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    protected readonly object SyncRoot = new object();
 | 
			
		||||
    private int m_receiveBufferSize = 1024 * 10;
 | 
			
		||||
    private int m_sendBufferSize = 1024 * 10;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public virtual int SendBufferSize
 | 
			
		||||
    {
 | 
			
		||||
        get => this.m_sendBufferSize;
 | 
			
		||||
        set => this.m_sendBufferSize = value < 1024 ? 1024 : value;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public virtual int ReceiveBufferSize
 | 
			
		||||
    {
 | 
			
		||||
        get => this.m_receiveBufferSize;
 | 
			
		||||
        set => this.m_receiveBufferSize = value < 1024 ? 1024 : value;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public ILog Logger { get; set; }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,396 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.Diagnostics;
 | 
			
		||||
using System.IO.Ports;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Serial;
 | 
			
		||||
internal sealed class InternalSerialCore : SerialCore
 | 
			
		||||
{
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// Serial核心
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class SerialCore : IDisposable, ISender
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 初始缓存大小
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public const int BufferSize = 1024 * 10;
 | 
			
		||||
    #region 字段
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 同步根
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public readonly object SyncRoot = new object();
 | 
			
		||||
 | 
			
		||||
    private long m_bufferRate;
 | 
			
		||||
    private bool m_disposedValue;
 | 
			
		||||
    private SpinLock m_lock;
 | 
			
		||||
    private bool m_online => MainSerialPort?.IsOpen == true;
 | 
			
		||||
    private int m_receiveBufferSize = BufferSize;
 | 
			
		||||
    private ValueCounter m_receiveCounter;
 | 
			
		||||
    private int m_sendBufferSize = BufferSize;
 | 
			
		||||
    private ValueCounter m_sendCounter;
 | 
			
		||||
    private readonly SemaphoreSlim m_semaphore = new SemaphoreSlim(1, 1);
 | 
			
		||||
    #endregion 字段
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Tcp核心
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public SerialCore()
 | 
			
		||||
    {
 | 
			
		||||
        this.m_lock = new SpinLock(Debugger.IsAttached);
 | 
			
		||||
        this.m_receiveCounter = new ValueCounter
 | 
			
		||||
        {
 | 
			
		||||
            Period = TimeSpan.FromSeconds(1),
 | 
			
		||||
            OnPeriod = this.OnReceivePeriod
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
        this.m_sendCounter = new ValueCounter
 | 
			
		||||
        {
 | 
			
		||||
            Period = TimeSpan.FromSeconds(1),
 | 
			
		||||
            OnPeriod = this.OnSendPeriod
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 析构函数
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    ~SerialCore()
 | 
			
		||||
    {
 | 
			
		||||
        this.Dispose(disposing: false);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public bool CanSend => this.m_online;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 当中断Tcp的时候。当为<see langword="true"/>时,意味着是调用<see cref="Close(string)"/>。当为<see langword="false"/>时,则是其他中断。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public Action<SerialCore, bool, string> OnBreakOut { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 当发生异常的时候
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public Action<SerialCore, Exception> OnException { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 在线状态
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool Online { get => this.m_online; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// UserToken
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public ByteBlock UserToken { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 当收到数据的时候
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public Action<SerialCore, ByteBlock> OnReceived { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 接收缓存池(可以设定初始值,运行时的值会根据流速自动调整)
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int ReceiveBufferSize
 | 
			
		||||
    {
 | 
			
		||||
        get => this.m_receiveBufferSize;
 | 
			
		||||
        set
 | 
			
		||||
        {
 | 
			
		||||
            this.m_receiveBufferSize = value;
 | 
			
		||||
            if (this.MainSerialPort != null && !MainSerialPort.IsOpen)
 | 
			
		||||
            {
 | 
			
		||||
                this.MainSerialPort.ReadBufferSize = value;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 接收计数器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public ValueCounter ReceiveCounter { get => this.m_receiveCounter; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 发送缓存池(可以设定初始值,运行时的值会根据流速自动调整)
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int SendBufferSize
 | 
			
		||||
    {
 | 
			
		||||
        get => this.m_sendBufferSize;
 | 
			
		||||
        set
 | 
			
		||||
        {
 | 
			
		||||
            this.m_sendBufferSize = value;
 | 
			
		||||
            if (this.MainSerialPort != null && !MainSerialPort.IsOpen)
 | 
			
		||||
            {
 | 
			
		||||
                this.MainSerialPort.WriteBufferSize = value;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 发送计数器
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public ValueCounter SendCounter { get => this.m_sendCounter; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// SerialPort
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public SerialPort MainSerialPort { get; private set; }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 开始以Iocp方式接收
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public virtual void BeginIocpReceive()
 | 
			
		||||
    {
 | 
			
		||||
        var byteBlock = BytePool.Default.GetByteBlock(this.ReceiveBufferSize);
 | 
			
		||||
        this.UserToken = byteBlock;
 | 
			
		||||
        byteBlock.SetLength(0);
 | 
			
		||||
        if (this.MainSerialPort.BytesToRead > 0)
 | 
			
		||||
        {
 | 
			
		||||
            this.ProcessReceived();
 | 
			
		||||
        }
 | 
			
		||||
        MainSerialPort.DataReceived += MainSerialPort_DataReceived;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void MainSerialPort_DataReceived(object sender, SerialDataReceivedEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            this.m_bufferRate = 1;
 | 
			
		||||
            this.ProcessReceived();
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            this.PrivateBreakOut(false, ex.Message);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 请求关闭
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="msg"></param>
 | 
			
		||||
    public virtual void Close(string msg)
 | 
			
		||||
    {
 | 
			
		||||
        this.PrivateBreakOut(true, msg);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 释放对象
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public void Dispose()
 | 
			
		||||
    {
 | 
			
		||||
        this.Dispose(disposing: true);
 | 
			
		||||
        GC.SuppressFinalize(this);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 重置环境,并设置新的<see cref="MainSerialPort"/>。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="socket"></param>
 | 
			
		||||
    public virtual void Reset(SerialPort socket)
 | 
			
		||||
    {
 | 
			
		||||
        if (socket is null)
 | 
			
		||||
        {
 | 
			
		||||
            throw new ArgumentNullException(nameof(socket));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (!socket.IsOpen)
 | 
			
		||||
        {
 | 
			
		||||
            throw new Exception("新的SerialPort必须在连接状态。");
 | 
			
		||||
        }
 | 
			
		||||
        this.Reset();
 | 
			
		||||
        this.MainSerialPort = socket;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 重置环境。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public virtual void Reset()
 | 
			
		||||
    {
 | 
			
		||||
        this.m_receiveCounter.Reset();
 | 
			
		||||
        this.m_sendCounter.Reset();
 | 
			
		||||
        this.MainSerialPort = null;
 | 
			
		||||
        this.OnReceived = null;
 | 
			
		||||
        this.OnBreakOut = null;
 | 
			
		||||
        this.UserToken = null;
 | 
			
		||||
        this.m_bufferRate = 1;
 | 
			
		||||
        this.m_lock = new SpinLock();
 | 
			
		||||
        this.m_receiveBufferSize = BufferSize;
 | 
			
		||||
        this.m_sendBufferSize = BufferSize;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 发送数据。
 | 
			
		||||
    /// <para>
 | 
			
		||||
    /// 内部会根据是否启用Ssl,进行直接发送,还是Ssl发送。
 | 
			
		||||
    /// </para>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="buffer"></param>
 | 
			
		||||
    /// <param name="offset"></param>
 | 
			
		||||
    /// <param name="length"></param>
 | 
			
		||||
    public virtual void Send(byte[] buffer, int offset, int length)
 | 
			
		||||
    {
 | 
			
		||||
        var lockTaken = false;
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            this.m_lock.Enter(ref lockTaken);
 | 
			
		||||
            this.MainSerialPort.Write(buffer, offset, length);
 | 
			
		||||
        }
 | 
			
		||||
        finally
 | 
			
		||||
        {
 | 
			
		||||
            if (lockTaken) this.m_lock.Exit(false);
 | 
			
		||||
        }
 | 
			
		||||
        this.m_sendCounter.Increment(length);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 异步发送数据。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="buffer"></param>
 | 
			
		||||
    /// <param name="offset"></param>
 | 
			
		||||
    /// <param name="length"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    /// <exception cref="Exception"></exception>
 | 
			
		||||
    public virtual async Task SendAsync(byte[] buffer, int offset, int length)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await this.m_semaphore.WaitAsync();
 | 
			
		||||
 | 
			
		||||
            this.MainSerialPort.Write(buffer, offset, length);
 | 
			
		||||
        }
 | 
			
		||||
        finally
 | 
			
		||||
        {
 | 
			
		||||
            this.m_semaphore.Release();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        this.m_sendCounter.Increment(length);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 当中断Tcp时。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="manual">当为<see langword="true"/>时,意味着是调用<see cref="Close(string)"/>。当为<see langword="false"/>时,则是其他中断。</param>
 | 
			
		||||
    /// <param name="msg"></param>
 | 
			
		||||
    protected virtual void BreakOut(bool manual, string msg)
 | 
			
		||||
    {
 | 
			
		||||
        this.OnBreakOut?.Invoke(this, manual, msg);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 释放对象
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="disposing"></param>
 | 
			
		||||
    protected virtual void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        if (!this.m_disposedValue)
 | 
			
		||||
        {
 | 
			
		||||
            if (disposing)
 | 
			
		||||
            {
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            this.m_disposedValue = true;
 | 
			
		||||
        }
 | 
			
		||||
        UserToken.SafeDispose();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 当发生异常的时候
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="ex"></param>
 | 
			
		||||
    protected virtual void Exception(Exception ex)
 | 
			
		||||
    {
 | 
			
		||||
        this.OnException?.Invoke(this, ex);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 当收到数据的时候
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="byteBlock"></param>
 | 
			
		||||
    protected virtual void Received(ByteBlock byteBlock)
 | 
			
		||||
    {
 | 
			
		||||
        this.OnReceived?.Invoke(this, byteBlock);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void HandleBuffer(ByteBlock byteBlock)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            this.m_receiveCounter.Increment(byteBlock.Length);
 | 
			
		||||
            this.Received(byteBlock);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            this.Exception(ex);
 | 
			
		||||
        }
 | 
			
		||||
        finally
 | 
			
		||||
        {
 | 
			
		||||
            byteBlock.Dispose();
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void OnReceivePeriod(long value)
 | 
			
		||||
    {
 | 
			
		||||
        this.ReceiveBufferSize = TouchSocketUtility.HitBufferLength(value);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void OnSendPeriod(long value)
 | 
			
		||||
    {
 | 
			
		||||
        this.SendBufferSize = TouchSocketUtility.HitBufferLength(value);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void PrivateBreakOut(bool manual, string msg)
 | 
			
		||||
    {
 | 
			
		||||
        lock (this.SyncRoot)
 | 
			
		||||
        {
 | 
			
		||||
            if (this.m_online)
 | 
			
		||||
            {
 | 
			
		||||
                this.BreakOut(manual, msg);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void ProcessReceived()
 | 
			
		||||
    {
 | 
			
		||||
        if (!this.m_online)
 | 
			
		||||
        {
 | 
			
		||||
            UserToken?.SafeDispose();
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        if (MainSerialPort.BytesToRead > 0)
 | 
			
		||||
        {
 | 
			
		||||
            var byteBlock = UserToken;
 | 
			
		||||
            byte[] buffer = BytePool.Default.Rent(MainSerialPort.BytesToRead);
 | 
			
		||||
            int num = MainSerialPort.Read(buffer, 0, MainSerialPort.BytesToRead);
 | 
			
		||||
            byteBlock.Write(buffer, 0, num);
 | 
			
		||||
            byteBlock.SetLength(num);
 | 
			
		||||
            this.HandleBuffer(byteBlock);
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                var newByteBlock = BytePool.Default.GetByteBlock((int)Math.Min(this.ReceiveBufferSize * this.m_bufferRate, TouchSocketUtility.MaxBufferLength));
 | 
			
		||||
                newByteBlock.SetLength(0);
 | 
			
		||||
                UserToken = newByteBlock;
 | 
			
		||||
 | 
			
		||||
                if (MainSerialPort.BytesToRead > 0)
 | 
			
		||||
                {
 | 
			
		||||
                    this.m_bufferRate += 2;
 | 
			
		||||
                    this.ProcessReceived();
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            catch (Exception ex)
 | 
			
		||||
            {
 | 
			
		||||
                this.PrivateBreakOut(false, ex.Message);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,53 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.ComponentModel;
 | 
			
		||||
using System.IO.Ports;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// 串口属性
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class SerialProperty
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// COM
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("COM口")]
 | 
			
		||||
    public string PortName { get; set; } = "COM1";
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 波特率
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("波特率")]
 | 
			
		||||
    public int BaudRate { get; set; } = 9600;
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 数据位
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("数据位")]
 | 
			
		||||
    public int DataBits { get; set; } = 8;
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 校验位
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("校验位")]
 | 
			
		||||
    public Parity Parity { get; set; } = Parity.None;
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 停止位
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    [Description("停止位")]
 | 
			
		||||
    public StopBits StopBits { get; set; } = StopBits.One;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override string ToString()
 | 
			
		||||
    {
 | 
			
		||||
        return $"{PortName}[{BaudRate},{DataBits},{StopBits},{Parity}]";
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,817 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.IO.Ports;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/// <inheritdoc cref="SerialSessionBase"/>
 | 
			
		||||
public class SerialSession : SerialSessionBase
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 接收到数据
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public ReceivedEventHandler<SerialSession> Received { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override async Task ReceivedData(ReceivedDataEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        if (this.Received != null)
 | 
			
		||||
        {
 | 
			
		||||
            await this.Received.Invoke(this, e);
 | 
			
		||||
            if (e.Handled)
 | 
			
		||||
            {
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        await base.ReceivedData(e);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// 串口管理
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class SerialSessionBase : BaseSerial, ISerialSession
 | 
			
		||||
{
 | 
			
		||||
    static readonly Protocol SerialPort = new("SerialSession");
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 构造函数
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public SerialSessionBase()
 | 
			
		||||
    {
 | 
			
		||||
        this.Protocol = SerialPort;
 | 
			
		||||
        this.m_serialCore = new InternalSerialCore();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    #region 变量
 | 
			
		||||
 | 
			
		||||
    private DelaySender m_delaySender;
 | 
			
		||||
    private bool m_online => MainSerialPort?.IsOpen == true;
 | 
			
		||||
    private readonly SemaphoreSlim m_semaphore = new SemaphoreSlim(1, 1);
 | 
			
		||||
    private readonly InternalSerialCore m_serialCore;
 | 
			
		||||
    #endregion 变量
 | 
			
		||||
 | 
			
		||||
    #region 事件
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public ConnectedEventHandler<ISerialSession> Connected { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public SerialConnectingEventHandler<ISerialSession> Connecting { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public DisconnectEventHandler<ISerialSessionBase> Disconnected { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public DisconnectEventHandler<ISerialSessionBase> Disconnecting { get; set; }
 | 
			
		||||
 | 
			
		||||
    private Task PrivateOnConnected(object o)
 | 
			
		||||
    {
 | 
			
		||||
        return this.OnConnected((ConnectedEventArgs)o);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 已经建立连接
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="e"></param>
 | 
			
		||||
    protected virtual async Task OnConnected(ConnectedEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (this.Connected != null)
 | 
			
		||||
            {
 | 
			
		||||
                await this.Connected.Invoke(this, e);
 | 
			
		||||
                if (e.Handled)
 | 
			
		||||
                {
 | 
			
		||||
                    return;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            await this.PluginsManager.RaiseAsync(nameof(ITcpConnectedPlugin.OnTcpConnected), this, e);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            this.Logger.Log(LogLevel.Error, this, $"在事件{nameof(this.Connected)}中发生错误。", ex);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private Task PrivateOnConnecting(SerialConnectingEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        if (this.CanSetDataHandlingAdapter)
 | 
			
		||||
        {
 | 
			
		||||
            this.SetDataHandlingAdapter(this.Config.GetValue(TouchSocketConfigExtension.TcpDataHandlingAdapterProperty).Invoke());
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return this.OnConnecting(e);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 准备连接的时候,此时并未建立连接
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="e"></param>
 | 
			
		||||
    protected virtual async Task OnConnecting(SerialConnectingEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (this.Connecting != null)
 | 
			
		||||
            {
 | 
			
		||||
                await this.Connecting.Invoke(this, e);
 | 
			
		||||
                if (e.Handled)
 | 
			
		||||
                {
 | 
			
		||||
                    return;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            await this.PluginsManager.RaiseAsync(nameof(ITcpConnectingPlugin.OnTcpConnecting), this, e);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            this.Logger.Log(LogLevel.Error, this, $"在事件{nameof(this.OnConnecting)}中发生错误。", ex);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private Task PrivateOnDisconnected(object obj)
 | 
			
		||||
    {
 | 
			
		||||
        this.m_receiver?.TryInputReceive(default, default);
 | 
			
		||||
        return this.OnDisconnected((DisconnectEventArgs)obj);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 断开连接。在客户端未设置连接状态时,不会触发
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="e"></param>
 | 
			
		||||
    protected virtual async Task OnDisconnected(DisconnectEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (this.Disconnected != null)
 | 
			
		||||
            {
 | 
			
		||||
                await this.Disconnected.Invoke(this, e).ConfigureAwait(false);
 | 
			
		||||
                if (e.Handled)
 | 
			
		||||
                {
 | 
			
		||||
                    return;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            await this.PluginsManager.RaiseAsync(nameof(ITcpDisconnectedPlugin.OnTcpDisconnected), this, e).ConfigureAwait(false);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            this.Logger.Log(LogLevel.Error, this, $"在事件{nameof(this.Disconnected)}中发生错误。", ex);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private Task PrivateOnDisconnecting(object obj)
 | 
			
		||||
    {
 | 
			
		||||
        return this.OnDisconnecting((DisconnectEventArgs)obj);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 即将断开连接(仅主动断开时有效)。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="e"></param>
 | 
			
		||||
    protected virtual async Task OnDisconnecting(DisconnectEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (this.Disconnecting != null)
 | 
			
		||||
            {
 | 
			
		||||
                await this.Disconnecting.Invoke(this, e).ConfigureAwait(false);
 | 
			
		||||
                if (e.Handled)
 | 
			
		||||
                {
 | 
			
		||||
                    return;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            await this.PluginsManager.RaiseAsync(nameof(ITcpDisconnectingPlugin.OnTcpDisconnecting), this, e).ConfigureAwait(false);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            this.Logger.Log(LogLevel.Error, this, $"在事件{nameof(this.Disconnecting)}中发生错误。", ex);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    #endregion 事件
 | 
			
		||||
 | 
			
		||||
    #region 属性
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public DateTime LastReceivedTime => this.GetTcpCore().ReceiveCounter.LastIncrement;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public DateTime LastSendTime => this.GetTcpCore().SendCounter.LastIncrement;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public IContainer Container { get; private set; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public virtual bool CanSetDataHandlingAdapter => true;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public TouchSocketConfig Config { get; private set; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public SingleStreamDataHandlingAdapter DataHandlingAdapter { get; private set; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public SerialProperty SerialProperty { get; private set; }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public SerialPort MainSerialPort { get; private set; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public bool Online { get => this.m_online; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public bool CanSend => this.m_online;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public IPluginsManager PluginsManager { get; private set; }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public Protocol Protocol { get; set; }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    #endregion 属性
 | 
			
		||||
 | 
			
		||||
    #region 断开操作
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public virtual void Close(string msg = TouchSocketCoreUtility.Empty)
 | 
			
		||||
    {
 | 
			
		||||
        lock (this.SyncRoot)
 | 
			
		||||
        {
 | 
			
		||||
            if (this.m_online)
 | 
			
		||||
            {
 | 
			
		||||
                Task.Factory.StartNew(this.PrivateOnDisconnecting, new DisconnectEventArgs(true, msg));
 | 
			
		||||
                this.MainSerialPort.TryClose();
 | 
			
		||||
                this.BreakOut(default, true, msg);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="disposing"></param>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        lock (this.SyncRoot)
 | 
			
		||||
        {
 | 
			
		||||
            if (this.m_online)
 | 
			
		||||
            {
 | 
			
		||||
                Task.Factory.StartNew(this.PrivateOnDisconnecting, new DisconnectEventArgs(true, $"{nameof(Dispose)}主动断开"));
 | 
			
		||||
                this.BreakOut(default, true, $"{nameof(Dispose)}主动断开");
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        base.Dispose(disposing);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    #endregion 断开操作
 | 
			
		||||
 | 
			
		||||
    #region Connect
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 打开串口
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    protected void Open()
 | 
			
		||||
    {
 | 
			
		||||
        lock (this.SyncRoot)
 | 
			
		||||
        {
 | 
			
		||||
            if (this.m_online)
 | 
			
		||||
            {
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
            if (this.DisposedValue)
 | 
			
		||||
            {
 | 
			
		||||
                throw new ObjectDisposedException(this.GetType().FullName);
 | 
			
		||||
            }
 | 
			
		||||
            if (this.Config == null)
 | 
			
		||||
            {
 | 
			
		||||
                throw new ArgumentNullException(nameof(this.Config), "配置文件不能为空。");
 | 
			
		||||
            }
 | 
			
		||||
            var serialProperty = this.Config.GetValue(SerialConfigExtension.SerialProperty) ?? throw new ArgumentNullException("串口配置不能为空。");
 | 
			
		||||
            this.MainSerialPort.SafeDispose();
 | 
			
		||||
            var serialPort = CreateSerial(serialProperty);
 | 
			
		||||
            this.PrivateOnConnecting(new(serialPort))
 | 
			
		||||
                .ConfigureAwait(false)
 | 
			
		||||
                .GetAwaiter()
 | 
			
		||||
                .GetResult();
 | 
			
		||||
 | 
			
		||||
            serialPort.Open();
 | 
			
		||||
 | 
			
		||||
            this.SetSerialPort(serialPort);
 | 
			
		||||
            this.BeginReceive();
 | 
			
		||||
 | 
			
		||||
            Task.Factory.StartNew(this.PrivateOnConnected, new ConnectedEventArgs());
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    private void BeginReceive()
 | 
			
		||||
    {
 | 
			
		||||
        this.GetTcpCore().BeginIocpReceive();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public virtual ISerialSession Connect()
 | 
			
		||||
    {
 | 
			
		||||
        this.Open();
 | 
			
		||||
        return this;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public Task<ISerialSession> ConnectAsync()
 | 
			
		||||
    {
 | 
			
		||||
        return Task.Run(() =>
 | 
			
		||||
        {
 | 
			
		||||
            return this.Connect();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    #endregion Connect
 | 
			
		||||
 | 
			
		||||
    #region Receiver
 | 
			
		||||
 | 
			
		||||
    private Receiver m_receiver;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public IReceiver CreateReceiver()
 | 
			
		||||
    {
 | 
			
		||||
        return this.m_receiver ??= new Receiver(this);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public void ClearReceiver()
 | 
			
		||||
    {
 | 
			
		||||
        this.m_receiver = null;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    #endregion
 | 
			
		||||
 | 
			
		||||
    private void BreakOut(SerialCore core, bool manual, string msg)
 | 
			
		||||
    {
 | 
			
		||||
        lock (this.SyncRoot)
 | 
			
		||||
        {
 | 
			
		||||
            if (this.m_online)
 | 
			
		||||
            {
 | 
			
		||||
                this.MainSerialPort.SafeDispose();
 | 
			
		||||
                this.m_delaySender.SafeDispose();
 | 
			
		||||
                this.DataHandlingAdapter.SafeDispose();
 | 
			
		||||
                Task.Factory.StartNew(this.PrivateOnDisconnected, new DisconnectEventArgs(manual, msg));
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private SerialCore GetTcpCore()
 | 
			
		||||
    {
 | 
			
		||||
        this.ThrowIfDisposed();
 | 
			
		||||
        return this.m_serialCore ?? throw new ObjectDisposedException(this.GetType().Name);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override int ReceiveBufferSize
 | 
			
		||||
    {
 | 
			
		||||
        get => this.GetTcpCore().ReceiveBufferSize;
 | 
			
		||||
        set
 | 
			
		||||
        {
 | 
			
		||||
            this.GetTcpCore().ReceiveBufferSize = value;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override int SendBufferSize
 | 
			
		||||
    {
 | 
			
		||||
        get => this.GetTcpCore().SendBufferSize;
 | 
			
		||||
        set
 | 
			
		||||
        {
 | 
			
		||||
            this.GetTcpCore().SendBufferSize = value;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public virtual void SetDataHandlingAdapter(SingleStreamDataHandlingAdapter adapter)
 | 
			
		||||
    {
 | 
			
		||||
        if (!this.CanSetDataHandlingAdapter)
 | 
			
		||||
        {
 | 
			
		||||
            throw new Exception($"不允许自由调用{nameof(SetDataHandlingAdapter)}进行赋值。");
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        this.SetAdapter(adapter);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public ISerialSession Setup(TouchSocketConfig config)
 | 
			
		||||
    {
 | 
			
		||||
        if (config == null)
 | 
			
		||||
        {
 | 
			
		||||
            throw new ArgumentNullException(nameof(config));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        this.ThrowIfDisposed();
 | 
			
		||||
 | 
			
		||||
        this.BuildConfig(config);
 | 
			
		||||
 | 
			
		||||
        this.PluginsManager.Raise(nameof(ILoadingConfigPlugin.OnLoadingConfig), this, new ConfigEventArgs(config));
 | 
			
		||||
        this.LoadConfig(this.Config);
 | 
			
		||||
        this.PluginsManager.Raise(nameof(ILoadedConfigPlugin.OnLoadedConfig), this, new ConfigEventArgs(config));
 | 
			
		||||
 | 
			
		||||
        return this;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void BuildConfig(TouchSocketConfig config)
 | 
			
		||||
    {
 | 
			
		||||
        this.Config = config;
 | 
			
		||||
 | 
			
		||||
        if (!(config.GetValue(TouchSocketCoreConfigExtension.ContainerProperty) is IContainer container))
 | 
			
		||||
        {
 | 
			
		||||
            container = new Container();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (!container.IsRegistered(typeof(ILog)))
 | 
			
		||||
        {
 | 
			
		||||
            container.RegisterSingleton<ILog, LoggerGroup>();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (!(config.GetValue(TouchSocketCoreConfigExtension.PluginsManagerProperty) is IPluginsManager pluginsManager))
 | 
			
		||||
        {
 | 
			
		||||
            pluginsManager = new PluginsManager(container);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (container.IsRegistered(typeof(IPluginsManager)))
 | 
			
		||||
        {
 | 
			
		||||
            pluginsManager = container.Resolve<IPluginsManager>();
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            container.RegisterSingleton<IPluginsManager>(pluginsManager);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (config.GetValue(TouchSocketCoreConfigExtension.ConfigureContainerProperty) is Action<IContainer> actionContainer)
 | 
			
		||||
        {
 | 
			
		||||
            actionContainer.Invoke(container);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (config.GetValue(TouchSocketCoreConfigExtension.ConfigurePluginsProperty) is Action<IPluginsManager> actionPluginsManager)
 | 
			
		||||
        {
 | 
			
		||||
            pluginsManager.Enable = true;
 | 
			
		||||
            actionPluginsManager.Invoke(pluginsManager);
 | 
			
		||||
        }
 | 
			
		||||
        this.Container = container;
 | 
			
		||||
        this.PluginsManager = pluginsManager;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void PrivateHandleReceivedData(ByteBlock byteBlock, IRequestInfo requestInfo)
 | 
			
		||||
    {
 | 
			
		||||
        if (this.m_receiver != null)
 | 
			
		||||
        {
 | 
			
		||||
            if (this.m_receiver.TryInputReceive(byteBlock, requestInfo))
 | 
			
		||||
            {
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        this.ReceivedData(new ReceivedDataEventArgs(byteBlock, requestInfo)).GetFalseAwaitResult();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 当收到适配器处理的数据时。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="e"></param>
 | 
			
		||||
    /// <returns>如果返回<see langword="true"/>则表示数据已被处理,且不会再向下传递。</returns>
 | 
			
		||||
    protected virtual Task ReceivedData(ReceivedDataEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
        return this.PluginsManager.RaiseAsync(nameof(ITcpReceivedPlugin.OnTcpReceived), this, e);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 当即将发送时,如果覆盖父类方法,则不会触发插件。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="buffer">数据缓存区</param>
 | 
			
		||||
    /// <param name="offset">偏移</param>
 | 
			
		||||
    /// <param name="length">长度</param>
 | 
			
		||||
    /// <returns>返回值表示是否允许发送</returns>
 | 
			
		||||
    protected virtual async Task<bool> SendingData(byte[] buffer, int offset, int length)
 | 
			
		||||
    {
 | 
			
		||||
        if (this.PluginsManager.GetPluginCount(nameof(ITcpSendingPlugin.OnTcpSending)) > 0)
 | 
			
		||||
        {
 | 
			
		||||
            var args = new SendingEventArgs(buffer, offset, length);
 | 
			
		||||
            await this.PluginsManager.RaiseAsync(nameof(ITcpSendingPlugin.OnTcpSending), this, args).ConfigureAwait(false);
 | 
			
		||||
            return args.IsPermitOperation;
 | 
			
		||||
        }
 | 
			
		||||
        return true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 加载配置
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="config"></param>
 | 
			
		||||
    protected virtual void LoadConfig(TouchSocketConfig config)
 | 
			
		||||
    {
 | 
			
		||||
        this.SerialProperty = config.GetValue(SerialConfigExtension.SerialProperty);
 | 
			
		||||
        this.Logger ??= this.Container.Resolve<ILog>();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 设置适配器,该方法不会检验<see cref="CanSetDataHandlingAdapter"/>的值。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="adapter"></param>
 | 
			
		||||
    protected void SetAdapter(SingleStreamDataHandlingAdapter adapter)
 | 
			
		||||
    {
 | 
			
		||||
        this.ThrowIfDisposed();
 | 
			
		||||
        if (adapter is null)
 | 
			
		||||
        {
 | 
			
		||||
            throw new ArgumentNullException(nameof(adapter));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (this.Config != null)
 | 
			
		||||
        {
 | 
			
		||||
            adapter.Config(this.Config);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        adapter.Logger = this.Logger;
 | 
			
		||||
        adapter.OnLoaded(this);
 | 
			
		||||
        adapter.ReceivedCallBack = this.PrivateHandleReceivedData;
 | 
			
		||||
        adapter.SendCallBack = this.DefaultSend;
 | 
			
		||||
        adapter.SendAsyncCallBack = this.DefaultSendAsync;
 | 
			
		||||
        this.DataHandlingAdapter = adapter;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private static SerialPort CreateSerial(SerialProperty serialProperty)
 | 
			
		||||
    {
 | 
			
		||||
        SerialPort serialPort = new(serialProperty.PortName, serialProperty.BaudRate, serialProperty.Parity, serialProperty.DataBits, serialProperty.StopBits)
 | 
			
		||||
        {
 | 
			
		||||
            DtrEnable = true,
 | 
			
		||||
            RtsEnable = true
 | 
			
		||||
        };
 | 
			
		||||
        return serialPort;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    #region 发送
 | 
			
		||||
 | 
			
		||||
    #region 同步发送
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="requestInfo"></param>
 | 
			
		||||
    /// <exception cref="NotConnectedException"></exception>
 | 
			
		||||
    /// <exception cref="OverlengthException"></exception>
 | 
			
		||||
    /// <exception cref="Exception"></exception>
 | 
			
		||||
    public void Send(IRequestInfo requestInfo)
 | 
			
		||||
    {
 | 
			
		||||
        if (this.DisposedValue)
 | 
			
		||||
        {
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        if (this.DataHandlingAdapter == null)
 | 
			
		||||
        {
 | 
			
		||||
            throw new ArgumentNullException(nameof(this.DataHandlingAdapter), TouchSocketResource.NullDataAdapter.GetDescription());
 | 
			
		||||
        }
 | 
			
		||||
        if (!this.DataHandlingAdapter.CanSendRequestInfo)
 | 
			
		||||
        {
 | 
			
		||||
            throw new NotSupportedException($"当前适配器不支持对象发送。");
 | 
			
		||||
        }
 | 
			
		||||
        this.DataHandlingAdapter.SendInput(requestInfo);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="buffer"><inheritdoc/></param>
 | 
			
		||||
    /// <param name="offset"><inheritdoc/></param>
 | 
			
		||||
    /// <param name="length"><inheritdoc/></param>
 | 
			
		||||
    /// <exception cref="NotConnectedException"><inheritdoc/></exception>
 | 
			
		||||
    /// <exception cref="OverlengthException"><inheritdoc/></exception>
 | 
			
		||||
    /// <exception cref="Exception"><inheritdoc/></exception>
 | 
			
		||||
    public virtual void Send(byte[] buffer, int offset, int length)
 | 
			
		||||
    {
 | 
			
		||||
        if (this.DataHandlingAdapter == null)
 | 
			
		||||
        {
 | 
			
		||||
            throw new ArgumentNullException(nameof(this.DataHandlingAdapter), TouchSocketResource.NullDataAdapter.GetDescription());
 | 
			
		||||
        }
 | 
			
		||||
        this.DataHandlingAdapter.SendInput(buffer, offset, length);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="transferBytes"><inheritdoc/></param>
 | 
			
		||||
    /// <exception cref="NotConnectedException"><inheritdoc/></exception>
 | 
			
		||||
    /// <exception cref="OverlengthException"><inheritdoc/></exception>
 | 
			
		||||
    /// <exception cref="Exception"><inheritdoc/></exception>
 | 
			
		||||
    public virtual void Send(IList<ArraySegment<byte>> transferBytes)
 | 
			
		||||
    {
 | 
			
		||||
        if (this.DataHandlingAdapter == null)
 | 
			
		||||
        {
 | 
			
		||||
            throw new ArgumentNullException(nameof(this.DataHandlingAdapter), TouchSocketResource.NullDataAdapter.GetDescription());
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (this.DataHandlingAdapter.CanSplicingSend)
 | 
			
		||||
        {
 | 
			
		||||
            this.DataHandlingAdapter.SendInput(transferBytes);
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            var length = 0;
 | 
			
		||||
            foreach (var item in transferBytes)
 | 
			
		||||
            {
 | 
			
		||||
                length += item.Count;
 | 
			
		||||
            }
 | 
			
		||||
            using (var byteBlock = new ByteBlock(length))
 | 
			
		||||
            {
 | 
			
		||||
                foreach (var item in transferBytes)
 | 
			
		||||
                {
 | 
			
		||||
                    byteBlock.Write(item.Array, item.Offset, item.Count);
 | 
			
		||||
                }
 | 
			
		||||
                this.DataHandlingAdapter.SendInput(byteBlock.Buffer, 0, byteBlock.Len);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    #endregion 同步发送
 | 
			
		||||
 | 
			
		||||
    #region 异步发送
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="buffer"></param>
 | 
			
		||||
    /// <param name="offset"></param>
 | 
			
		||||
    /// <param name="length"></param>
 | 
			
		||||
    /// <exception cref="NotConnectedException"></exception>
 | 
			
		||||
    /// <exception cref="OverlengthException"></exception>
 | 
			
		||||
    /// <exception cref="Exception"></exception>
 | 
			
		||||
    public virtual Task SendAsync(byte[] buffer, int offset, int length)
 | 
			
		||||
    {
 | 
			
		||||
        this.ThrowIfDisposed();
 | 
			
		||||
        if (this.DataHandlingAdapter == null)
 | 
			
		||||
        {
 | 
			
		||||
            throw new ArgumentNullException(nameof(this.DataHandlingAdapter), TouchSocketResource.NullDataAdapter.GetDescription());
 | 
			
		||||
        }
 | 
			
		||||
        return this.DataHandlingAdapter.SendInputAsync(buffer, offset, length);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="requestInfo"></param>
 | 
			
		||||
    /// <exception cref="NotConnectedException"></exception>
 | 
			
		||||
    /// <exception cref="OverlengthException"></exception>
 | 
			
		||||
    /// <exception cref="Exception"></exception>
 | 
			
		||||
    public virtual Task SendAsync(IRequestInfo requestInfo)
 | 
			
		||||
    {
 | 
			
		||||
        this.ThrowIfDisposed();
 | 
			
		||||
        if (this.DataHandlingAdapter == null)
 | 
			
		||||
        {
 | 
			
		||||
            throw new ArgumentNullException(nameof(this.DataHandlingAdapter), TouchSocketResource.NullDataAdapter.GetDescription());
 | 
			
		||||
        }
 | 
			
		||||
        if (!this.DataHandlingAdapter.CanSendRequestInfo)
 | 
			
		||||
        {
 | 
			
		||||
            throw new NotSupportedException($"当前适配器不支持对象发送。");
 | 
			
		||||
        }
 | 
			
		||||
        return this.DataHandlingAdapter.SendInputAsync(requestInfo);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="transferBytes"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    /// <exception cref="ArgumentNullException"></exception>
 | 
			
		||||
    public virtual Task SendAsync(IList<ArraySegment<byte>> transferBytes)
 | 
			
		||||
    {
 | 
			
		||||
        this.ThrowIfDisposed();
 | 
			
		||||
        if (this.DataHandlingAdapter == null)
 | 
			
		||||
        {
 | 
			
		||||
            throw new ArgumentNullException(nameof(this.DataHandlingAdapter), TouchSocketResource.NullDataAdapter.GetDescription());
 | 
			
		||||
        }
 | 
			
		||||
        if (this.DataHandlingAdapter.CanSplicingSend)
 | 
			
		||||
        {
 | 
			
		||||
            return this.DataHandlingAdapter.SendInputAsync(transferBytes);
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            var length = 0;
 | 
			
		||||
            foreach (var item in transferBytes)
 | 
			
		||||
            {
 | 
			
		||||
                length += item.Count;
 | 
			
		||||
            }
 | 
			
		||||
            using (var byteBlock = new ByteBlock(length))
 | 
			
		||||
            {
 | 
			
		||||
                foreach (var item in transferBytes)
 | 
			
		||||
                {
 | 
			
		||||
                    byteBlock.Write(item.Array, item.Offset, item.Count);
 | 
			
		||||
                }
 | 
			
		||||
                return this.DataHandlingAdapter.SendInputAsync(byteBlock.Buffer, 0, byteBlock.Len);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    #endregion 异步发送
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public void DefaultSend(byte[] buffer, int offset, int length)
 | 
			
		||||
    {
 | 
			
		||||
        if (this.SendingData(buffer, offset, length).GetFalseAwaitResult())
 | 
			
		||||
        {
 | 
			
		||||
            this.GetTcpCore().Send(buffer, offset, length);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public async Task DefaultSendAsync(byte[] buffer, int offset, int length)
 | 
			
		||||
    {
 | 
			
		||||
        if (await this.SendingData(buffer, offset, length))
 | 
			
		||||
        {
 | 
			
		||||
            await this.GetTcpCore().SendAsync(buffer, offset, length);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    #endregion 发送
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    #region 自定义
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    private void SetSerialPort(SerialPort serialPort)
 | 
			
		||||
    {
 | 
			
		||||
        if (serialPort == null)
 | 
			
		||||
        {
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        this.MainSerialPort = serialPort;
 | 
			
		||||
        this.SerialProperty ??= new();
 | 
			
		||||
        this.SerialProperty.Parity = serialPort.Parity;
 | 
			
		||||
        this.SerialProperty.PortName = serialPort.PortName;
 | 
			
		||||
        this.SerialProperty.StopBits = serialPort.StopBits;
 | 
			
		||||
        this.SerialProperty.DataBits = serialPort.DataBits;
 | 
			
		||||
        this.SerialProperty.BaudRate = serialPort.BaudRate;
 | 
			
		||||
 | 
			
		||||
        var delaySenderOption = this.Config.GetValue(TouchSocketConfigExtension.DelaySenderProperty);
 | 
			
		||||
        if (delaySenderOption != null)
 | 
			
		||||
        {
 | 
			
		||||
            this.m_delaySender = new DelaySender(delaySenderOption, this.MainSerialPort.AbsoluteSend);
 | 
			
		||||
        }
 | 
			
		||||
        this.m_serialCore.Reset(serialPort);
 | 
			
		||||
        this.m_serialCore.OnReceived = this.HandleReceived;
 | 
			
		||||
        this.m_serialCore.OnBreakOut = this.BreakOut;
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void HandleReceived(SerialCore core, ByteBlock byteBlock)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (this.DisposedValue)
 | 
			
		||||
            {
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
            if (this.ReceivingData(byteBlock).GetFalseAwaitResult())
 | 
			
		||||
            {
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (this.DataHandlingAdapter == null)
 | 
			
		||||
            {
 | 
			
		||||
                this.Logger.Error(this, TouchSocketResource.NullDataAdapter.GetDescription());
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
            this.DataHandlingAdapter.ReceivedInput(byteBlock);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            this.Logger.Log(LogLevel.Error, this, "在处理数据时发生错误", ex);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 当收到原始数据
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="byteBlock"></param>
 | 
			
		||||
    /// <returns>如果返回<see langword="true"/>则表示数据已被处理,且不会再向下传递。</returns>
 | 
			
		||||
    protected virtual Task<bool> ReceivingData(ByteBlock byteBlock)
 | 
			
		||||
    {
 | 
			
		||||
        if (this.PluginsManager.GetPluginCount(nameof(ITcpReceivingPlugin.OnTcpReceiving)) > 0)
 | 
			
		||||
        {
 | 
			
		||||
            return this.PluginsManager.RaiseAsync(nameof(ITcpReceivingPlugin.OnTcpReceiving), this, new ByteBlockEventArgs(byteBlock));
 | 
			
		||||
        }
 | 
			
		||||
        return Task.FromResult(false);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    #endregion
 | 
			
		||||
}
 | 
			
		||||
@@ -1,58 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权(除特别声明或在XREF结尾的命名空间的代码)归作者本人若汝棋茗所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议,若本仓库没有设置,则按MIT开源协议授权
 | 
			
		||||
//  CSDN博客:https://blog.csdn.net/qq_40374647
 | 
			
		||||
//  哔哩哔哩视频:https://space.bilibili.com/94253567
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/RRQM_Home
 | 
			
		||||
//  Github源代码仓库:https://github.com/RRQM
 | 
			
		||||
//  API首页:http://rrqm_home.gitee.io/touchsocket/
 | 
			
		||||
//  交流QQ群:234762506
 | 
			
		||||
//  感谢您的下载和使用
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Sockets
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 通讯基类
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public abstract class BaseSocket : DependencyObject, ISocket
 | 
			
		||||
    {
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 同步根。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        protected readonly object SyncRoot = new object();
 | 
			
		||||
 | 
			
		||||
        private int m_receiveBufferSize = 1024 * 10;
 | 
			
		||||
        private int m_sendBufferSize = 1024 * 10;
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public virtual int SendBufferSize
 | 
			
		||||
        {
 | 
			
		||||
            get => this.m_sendBufferSize;
 | 
			
		||||
            set => this.m_sendBufferSize = value < 1024 ? 1024 : value;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public virtual int ReceiveBufferSize
 | 
			
		||||
        {
 | 
			
		||||
            get => this.m_receiveBufferSize;
 | 
			
		||||
            set => this.m_receiveBufferSize = value < 1024 ? 1024 : value;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public ILog Logger { get; set; }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,109 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权(除特别声明或在XREF结尾的命名空间的代码)归作者本人若汝棋茗所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议,若本仓库没有设置,则按MIT开源协议授权
 | 
			
		||||
//  CSDN博客:https://blog.csdn.net/qq_40374647
 | 
			
		||||
//  哔哩哔哩视频:https://space.bilibili.com/94253567
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/RRQM_Home
 | 
			
		||||
//  Github源代码仓库:https://github.com/RRQM
 | 
			
		||||
//  API首页:http://rrqm_home.gitee.io/touchsocket/
 | 
			
		||||
//  交流QQ群:234762506
 | 
			
		||||
//  感谢您的下载和使用
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Sockets
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 发送等待接口
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public interface IWaitSender : ISenderBase
 | 
			
		||||
    {
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 发送字节流
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="buffer">数据缓存区</param>
 | 
			
		||||
        /// <param name="offset">偏移</param>
 | 
			
		||||
        /// <param name="length">长度</param>
 | 
			
		||||
        /// <param name="timeout">超时时间</param>
 | 
			
		||||
        /// <param name="cancellationToken">取消令箭</param>
 | 
			
		||||
        /// <exception cref="NotConnectedException">客户端没有连接</exception>
 | 
			
		||||
        /// <exception cref="OverlengthException">发送数据超长</exception>
 | 
			
		||||
        /// <exception cref="Exception">其他异常</exception>
 | 
			
		||||
        /// <returns>返回的数据</returns>
 | 
			
		||||
        byte[] SendThenReturn(byte[] buffer, int offset, int length, int timeout = 1000 * 5, CancellationToken cancellationToken = default);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 发送字节流
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="buffer">数据缓存区</param>
 | 
			
		||||
        /// <param name="timeout">超时时间</param>
 | 
			
		||||
        /// <param name="cancellationToken">取消令箭</param>
 | 
			
		||||
        /// <exception cref="NotConnectedException">客户端没有连接</exception>
 | 
			
		||||
        /// <exception cref="OverlengthException">发送数据超长</exception>
 | 
			
		||||
        /// <exception cref="Exception">其他异常</exception>
 | 
			
		||||
        /// <returns>返回的数据</returns>
 | 
			
		||||
        byte[] SendThenReturn(byte[] buffer, int timeout = 1000 * 5, CancellationToken cancellationToken = default);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 发送流中的有效数据
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="byteBlock">数据块载体</param>
 | 
			
		||||
        /// <param name="timeout">超时时间</param>
 | 
			
		||||
        /// <param name="cancellationToken">取消令箭</param>
 | 
			
		||||
        /// <exception cref="NotConnectedException">客户端没有连接</exception>
 | 
			
		||||
        /// <exception cref="OverlengthException">发送数据超长</exception>
 | 
			
		||||
        /// <exception cref="Exception">其他异常</exception>
 | 
			
		||||
        /// <returns>返回的数据</returns>
 | 
			
		||||
        byte[] SendThenReturn(ByteBlock byteBlock, int timeout = 1000 * 5, CancellationToken cancellationToken = default);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 异步发送
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="buffer">数据缓存区</param>
 | 
			
		||||
        /// <param name="offset">偏移</param>
 | 
			
		||||
        /// <param name="length">长度</param>
 | 
			
		||||
        /// <param name="timeout">超时时间</param>
 | 
			
		||||
        /// <param name="cancellationToken">取消令箭</param>
 | 
			
		||||
        /// <exception cref="NotConnectedException">客户端没有连接</exception>
 | 
			
		||||
        /// <exception cref="OverlengthException">发送数据超长</exception>
 | 
			
		||||
        /// <exception cref="Exception">其他异常</exception>
 | 
			
		||||
        /// <returns>返回的数据</returns>
 | 
			
		||||
        Task<byte[]> SendThenReturnAsync(byte[] buffer, int offset, int length, int timeout = 1000 * 5, CancellationToken cancellationToken = default);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 异步发送
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="buffer">数据缓存区</param>
 | 
			
		||||
        /// <param name="timeout">超时时间</param>
 | 
			
		||||
        /// <param name="cancellationToken">取消令箭</param>
 | 
			
		||||
        /// <exception cref="NotConnectedException">客户端没有连接</exception>
 | 
			
		||||
        /// <exception cref="OverlengthException">发送数据超长</exception>
 | 
			
		||||
        /// <exception cref="Exception">其他异常</exception>
 | 
			
		||||
        /// <returns>返回的数据</returns>
 | 
			
		||||
        Task<byte[]> SendThenReturnAsync(byte[] buffer, int timeout = 1000 * 5, CancellationToken cancellationToken = default);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 异步发送
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="byteBlock">数据块载体</param>
 | 
			
		||||
        /// <param name="timeout">超时时间</param>
 | 
			
		||||
        /// <param name="cancellationToken">取消令箭</param>
 | 
			
		||||
        /// <exception cref="NotConnectedException">客户端没有连接</exception>
 | 
			
		||||
        /// <exception cref="OverlengthException">发送数据超长</exception>
 | 
			
		||||
        /// <exception cref="Exception">其他异常</exception>
 | 
			
		||||
        /// <returns>返回的数据</returns>
 | 
			
		||||
        Task<byte[]> SendThenReturnAsync(ByteBlock byteBlock, int timeout = 1000 * 5, CancellationToken cancellationToken = default);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,119 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权(除特别声明或在XREF结尾的命名空间的代码)归作者本人若汝棋茗所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议,若本仓库没有设置,则按MIT开源协议授权
 | 
			
		||||
//  CSDN博客:https://blog.csdn.net/qq_40374647
 | 
			
		||||
//  哔哩哔哩视频:https://space.bilibili.com/94253567
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/RRQM_Home
 | 
			
		||||
//  Github源代码仓库:https://github.com/RRQM
 | 
			
		||||
//  API首页:http://rrqm_home.gitee.io/touchsocket/
 | 
			
		||||
//  交流QQ群:234762506
 | 
			
		||||
//  感谢您的下载和使用
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Sockets
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 等待型客户端。
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public interface IWaitingClient<TClient> : IWaitSender, IDisposable where TClient : IClient, ISender
 | 
			
		||||
    {
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 等待设置。
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public WaitingOptions WaitingOptions { get; }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 客户端终端
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        TClient Client { get; }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 发送字节流
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="buffer">数据缓存区</param>
 | 
			
		||||
        /// <param name="offset">偏移</param>
 | 
			
		||||
        /// <param name="length">长度</param>
 | 
			
		||||
        /// <param name="timeout">超时时间</param>
 | 
			
		||||
        /// <param name="cancellationToken">取消令箭</param>
 | 
			
		||||
        /// <exception cref="NotConnectedException">客户端没有连接</exception>
 | 
			
		||||
        /// <exception cref="OverlengthException">发送数据超长</exception>
 | 
			
		||||
        /// <exception cref="Exception">其他异常</exception>
 | 
			
		||||
        /// <returns>返回的数据</returns>
 | 
			
		||||
        ResponsedData SendThenResponse(byte[] buffer, int offset, int length, int timeout = 1000 * 5, CancellationToken cancellationToken = default);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 发送字节流
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="buffer">数据缓存区</param>
 | 
			
		||||
        /// <param name="timeout">超时时间</param>
 | 
			
		||||
        /// <param name="cancellationToken">取消令箭</param>
 | 
			
		||||
        /// <exception cref="NotConnectedException">客户端没有连接</exception>
 | 
			
		||||
        /// <exception cref="OverlengthException">发送数据超长</exception>
 | 
			
		||||
        /// <exception cref="Exception">其他异常</exception>
 | 
			
		||||
        /// <returns>返回的数据</returns>
 | 
			
		||||
        ResponsedData SendThenResponse(byte[] buffer, int timeout = 1000 * 5, CancellationToken cancellationToken = default);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 发送流中的有效数据
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="byteBlock">数据块载体</param>
 | 
			
		||||
        /// <param name="timeout">超时时间</param>
 | 
			
		||||
        /// <param name="cancellationToken">取消令箭</param>
 | 
			
		||||
        /// <exception cref="NotConnectedException">客户端没有连接</exception>
 | 
			
		||||
        /// <exception cref="OverlengthException">发送数据超长</exception>
 | 
			
		||||
        /// <exception cref="Exception">其他异常</exception>
 | 
			
		||||
        /// <returns>返回的数据</returns>
 | 
			
		||||
        ResponsedData SendThenResponse(ByteBlock byteBlock, int timeout = 1000 * 5, CancellationToken cancellationToken = default);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 异步发送
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="buffer">数据缓存区</param>
 | 
			
		||||
        /// <param name="offset">偏移</param>
 | 
			
		||||
        /// <param name="length">长度</param>
 | 
			
		||||
        /// <param name="timeout">超时时间</param>
 | 
			
		||||
        /// <param name="cancellationToken">取消令箭</param>
 | 
			
		||||
        /// <exception cref="NotConnectedException">客户端没有连接</exception>
 | 
			
		||||
        /// <exception cref="OverlengthException">发送数据超长</exception>
 | 
			
		||||
        /// <exception cref="Exception">其他异常</exception>
 | 
			
		||||
        /// <returns>返回的数据</returns>
 | 
			
		||||
        Task<ResponsedData> SendThenResponseAsync(byte[] buffer, int offset, int length, int timeout = 1000 * 5, CancellationToken cancellationToken = default);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 异步发送
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="buffer">数据缓存区</param>
 | 
			
		||||
        /// <param name="timeout">超时时间</param>
 | 
			
		||||
        /// <param name="cancellationToken">取消令箭</param>
 | 
			
		||||
        /// <exception cref="NotConnectedException">客户端没有连接</exception>
 | 
			
		||||
        /// <exception cref="OverlengthException">发送数据超长</exception>
 | 
			
		||||
        /// <exception cref="Exception">其他异常</exception>
 | 
			
		||||
        /// <returns>返回的数据</returns>
 | 
			
		||||
        Task<ResponsedData> SendThenResponseAsync(byte[] buffer, int timeout = 1000 * 5, CancellationToken cancellationToken = default);
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 异步发送
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="byteBlock">数据块载体</param>
 | 
			
		||||
        /// <param name="timeout">超时时间</param>
 | 
			
		||||
        /// <param name="cancellationToken">取消令箭</param>
 | 
			
		||||
        /// <exception cref="NotConnectedException">客户端没有连接</exception>
 | 
			
		||||
        /// <exception cref="OverlengthException">发送数据超长</exception>
 | 
			
		||||
        /// <exception cref="Exception">其他异常</exception>
 | 
			
		||||
        /// <returns>返回的数据</returns>
 | 
			
		||||
        Task<ResponsedData> SendThenResponseAsync(ByteBlock byteBlock, int timeout = 1000 * 5, CancellationToken cancellationToken = default);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,29 +0,0 @@
 | 
			
		||||
<Project>
 | 
			
		||||
	<PropertyGroup>
 | 
			
		||||
		<Version>3.0.0.12</Version>
 | 
			
		||||
		<LangVersion>latest</LangVersion>
 | 
			
		||||
		<ImplicitUsings>enable</ImplicitUsings>
 | 
			
		||||
		<TargetFrameworks>net6.0;net7.0</TargetFrameworks>
 | 
			
		||||
		<Authors>Diego</Authors>
 | 
			
		||||
		<Product>ThingsGateway</Product>
 | 
			
		||||
		<Copyright>© 2023-present Diego</Copyright>
 | 
			
		||||
		<RepositoryUrl>https://gitee.com/diego2098/ThingsGateway</RepositoryUrl>
 | 
			
		||||
		<SignAssembly>True</SignAssembly>
 | 
			
		||||
		<DelaySign>False</DelaySign>
 | 
			
		||||
		<SatelliteResourceLanguages>zh-Hans</SatelliteResourceLanguages>
 | 
			
		||||
 | 
			
		||||
	</PropertyGroup>
 | 
			
		||||
 | 
			
		||||
	<!--<PropertyGroup>
 | 
			
		||||
		<EnableDynamicLoading>true</EnableDynamicLoading>
 | 
			
		||||
	</PropertyGroup>-->
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
 | 
			
		||||
		<ProjectReference Include="..\..\Web\ThingsGateway.Gateway.Application\ThingsGateway.Gateway.Application.csproj">
 | 
			
		||||
			<Private>false</Private>
 | 
			
		||||
			<CopyLocalSatelliteAssemblies>false</CopyLocalSatelliteAssemblies>
 | 
			
		||||
			<ExcludeAssets>runtime</ExcludeAssets>
 | 
			
		||||
		</ProjectReference>
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
</Project>
 | 
			
		||||
@@ -1,111 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.DLT645;
 | 
			
		||||
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
public class DLT645_2007 : CollectBase
 | 
			
		||||
{
 | 
			
		||||
    private readonly DLT645_2007Property driverPropertys = new();
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.DLT645.DLT645_2007 _plc;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => typeof(DLT645_2007DebugPage);
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public override bool IsSupportRequest => true;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override IThingsGatewayBitConverter ThingsGatewayBitConverter { get => _plc?.ThingsGatewayBitConverter; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override IReadWrite PLC => _plc;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        await _plc?.ConnectAsync(cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void InitDataAdapter()
 | 
			
		||||
    {
 | 
			
		||||
        _plc.SetDataAdapter();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsConnected()
 | 
			
		||||
    {
 | 
			
		||||
        return _plc?.SerialSession?.CanSend == true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableSourceRead> LoadSourceRead(List<DeviceVariableRunTime> deviceVariables)
 | 
			
		||||
    {
 | 
			
		||||
        return _plc.LoadSourceRead<DeviceVariableSourceRead, DeviceVariableRunTime>(deviceVariables, 0);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        _plc?.SafeDispose();
 | 
			
		||||
        base.Dispose(disposing);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
    {
 | 
			
		||||
        if (client == null)
 | 
			
		||||
        {
 | 
			
		||||
            FoundataionConfig.SetSerialProperty(new()
 | 
			
		||||
            {
 | 
			
		||||
                PortName = driverPropertys.PortName,
 | 
			
		||||
                BaudRate = driverPropertys.BaudRate,
 | 
			
		||||
                DataBits = driverPropertys.DataBits,
 | 
			
		||||
                Parity = driverPropertys.Parity,
 | 
			
		||||
                StopBits = driverPropertys.StopBits,
 | 
			
		||||
            })
 | 
			
		||||
                ;
 | 
			
		||||
            client = new SerialSession();
 | 
			
		||||
            ((SerialSession)client).Setup(FoundataionConfig);
 | 
			
		||||
        }
 | 
			
		||||
        //载入配置
 | 
			
		||||
        _plc = new((SerialSession)client)
 | 
			
		||||
        {
 | 
			
		||||
            FrameTime = driverPropertys.FrameTime,
 | 
			
		||||
            CacheTimeout = driverPropertys.CacheTimeout,
 | 
			
		||||
            DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
            EnableFEHead = driverPropertys.EnableFEHead,
 | 
			
		||||
            OperCode = driverPropertys.OperCode,
 | 
			
		||||
            Password = driverPropertys.Password,
 | 
			
		||||
            Station = driverPropertys.Station,
 | 
			
		||||
            TimeOut = driverPropertys.TimeOut
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override async Task<OperResult<byte[]>> ReadAsync(string address, int length, CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return await _plc.ReadAsync(address, length, cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,103 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.DLT645;
 | 
			
		||||
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
public class DLT645_2007OverTcp : CollectBase
 | 
			
		||||
{
 | 
			
		||||
 | 
			
		||||
    private readonly DLT645_2007OverTcpProperty driverPropertys = new();
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.DLT645.DLT645_2007OverTcp _plc;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => typeof(DLT645_2007OverTcpDebugPage);
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsSupportRequest => true;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override IThingsGatewayBitConverter ThingsGatewayBitConverter { get => _plc?.ThingsGatewayBitConverter; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override IReadWrite PLC => _plc;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        await _plc?.ConnectAsync(cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void InitDataAdapter()
 | 
			
		||||
    {
 | 
			
		||||
        _plc.SetDataAdapter();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsConnected()
 | 
			
		||||
    {
 | 
			
		||||
        return _plc?.TcpClient?.CanSend == true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableSourceRead> LoadSourceRead(List<DeviceVariableRunTime> deviceVariables)
 | 
			
		||||
    {
 | 
			
		||||
        return _plc.LoadSourceRead<DeviceVariableSourceRead, DeviceVariableRunTime>(deviceVariables, 0);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        base.Dispose(disposing);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
    {
 | 
			
		||||
        if (client == null)
 | 
			
		||||
        {
 | 
			
		||||
            FoundataionConfig.SetRemoteIPHost(new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}"))
 | 
			
		||||
                ;
 | 
			
		||||
            client = new TcpClient();
 | 
			
		||||
            ((TcpClient)client).Setup(FoundataionConfig);
 | 
			
		||||
        }
 | 
			
		||||
        //载入配置
 | 
			
		||||
        _plc = new((TcpClient)client)
 | 
			
		||||
        {
 | 
			
		||||
            FrameTime = driverPropertys.FrameTime,
 | 
			
		||||
            CacheTimeout = driverPropertys.CacheTimeout,
 | 
			
		||||
            DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
            EnableFEHead = driverPropertys.EnableFEHead,
 | 
			
		||||
            OperCode = driverPropertys.OperCode,
 | 
			
		||||
            Password = driverPropertys.Password,
 | 
			
		||||
            Station = driverPropertys.Station,
 | 
			
		||||
            TimeOut = driverPropertys.TimeOut
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override async Task<OperResult<byte[]>> ReadAsync(string address, int length, CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return await _plc.ReadAsync(address, length, cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,23 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
global using System;
 | 
			
		||||
global using System.Collections.Generic;
 | 
			
		||||
global using System.Threading;
 | 
			
		||||
global using System.Threading.Tasks;
 | 
			
		||||
 | 
			
		||||
global using ThingsGateway.Foundation.Core;
 | 
			
		||||
global using ThingsGateway.Foundation.Demo;
 | 
			
		||||
global using ThingsGateway.Foundation.Serial;
 | 
			
		||||
global using ThingsGateway.Foundation.Sockets;
 | 
			
		||||
global using ThingsGateway.Gateway.Application;
 | 
			
		||||
global using ThingsGateway.Gateway.Core;
 | 
			
		||||
@@ -1,35 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@using System.Net.Http
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Forms
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Routing
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web.Virtualization
 | 
			
		||||
@using Microsoft.JSInterop
 | 
			
		||||
@using BlazorComponent
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
@using Masa.Blazor.Presets
 | 
			
		||||
@using System.Net.Http.Json
 | 
			
		||||
@using System.IO;
 | 
			
		||||
@using System.Text.Json;
 | 
			
		||||
@using System.Reflection;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Components;
 | 
			
		||||
@using ThingsGateway.Admin.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Extension.String;
 | 
			
		||||
@using ThingsGateway.Admin.Application;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Core;
 | 
			
		||||
@using ThingsGateway.Gateway.Application;
 | 
			
		||||
@using ThingsGateway.Gateway.Core;
 | 
			
		||||
@@ -1,413 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using Confluent.Kafka;
 | 
			
		||||
 | 
			
		||||
using Mapster;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Kafka;
 | 
			
		||||
 | 
			
		||||
using Furion;
 | 
			
		||||
 | 
			
		||||
using System;
 | 
			
		||||
using System.Collections.Concurrent;
 | 
			
		||||
using System.Collections.Generic;
 | 
			
		||||
using System.Linq;
 | 
			
		||||
using System.Runtime.InteropServices;
 | 
			
		||||
using System.Threading;
 | 
			
		||||
using System.Threading.Tasks;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Extension.ConcurrentQueue;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.Generic;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.String;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// Kafka消息生产
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class KafkaProducer : UpLoadBase
 | 
			
		||||
{
 | 
			
		||||
    private readonly ConcurrentQueue<DeviceData> _collectDeviceRunTimes = new();
 | 
			
		||||
    private readonly ConcurrentQueue<VariableData> _collectVariableRunTimes = new();
 | 
			
		||||
    private readonly KafkaProducerProperty driverPropertys = new();
 | 
			
		||||
    private readonly KafkaProducerVariableProperty variablePropertys = new();
 | 
			
		||||
    private GlobalDeviceData _globalDeviceData;
 | 
			
		||||
    private List<DeviceVariableRunTime> _uploadVariables = new();
 | 
			
		||||
    private bool isSuccess = true;
 | 
			
		||||
    private IProducer<Null, string> producer;
 | 
			
		||||
    private ProducerBuilder<Null, string> producerBuilder;
 | 
			
		||||
    private ProducerConfig producerconfig;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => null;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override DriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
    private TimerTick exVariableTimerTick;
 | 
			
		||||
    private TimerTick exDeviceTimerTick;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableRunTime> UploadVariables => _uploadVariables;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override VariablePropertyBase VariablePropertys => variablePropertys;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task ExecuteAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (!driverPropertys.IsInterval)
 | 
			
		||||
            {
 | 
			
		||||
                ////变化推送
 | 
			
		||||
                var varList = _collectVariableRunTimes.ToListWithDequeue();
 | 
			
		||||
                if (varList?.Count != 0)
 | 
			
		||||
                {
 | 
			
		||||
                    //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
                    var varData = varList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                    foreach (var item in varData)
 | 
			
		||||
                    {
 | 
			
		||||
                        try
 | 
			
		||||
                        {
 | 
			
		||||
                            if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                            {
 | 
			
		||||
                                await KafKaUp(driverPropertys.VariableTopic, item.ToJsonString(), cancellationToken);
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                break;
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        catch (Exception ex)
 | 
			
		||||
                        {
 | 
			
		||||
                            LogMessage.LogWarning(ex, ToString());
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                    }
 | 
			
		||||
                    if (isSuccess)
 | 
			
		||||
                        producer.Flush(cancellationToken);
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                if (exVariableTimerTick.IsTickHappen())
 | 
			
		||||
                {
 | 
			
		||||
                    try
 | 
			
		||||
                    {
 | 
			
		||||
                        var varList = _uploadVariables.Adapt<List<VariableData>>();
 | 
			
		||||
                        if (varList?.Count != 0)
 | 
			
		||||
                        {
 | 
			
		||||
                            //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
                            var varData = varList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                            foreach (var item in varData)
 | 
			
		||||
                            {
 | 
			
		||||
                                try
 | 
			
		||||
                                {
 | 
			
		||||
                                    if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                                    {
 | 
			
		||||
                                        await KafKaUp(driverPropertys.VariableTopic, item.ToJsonString(), cancellationToken);
 | 
			
		||||
                                    }
 | 
			
		||||
                                    else
 | 
			
		||||
                                    {
 | 
			
		||||
                                        break;
 | 
			
		||||
                                    }
 | 
			
		||||
                                }
 | 
			
		||||
                                catch (Exception ex)
 | 
			
		||||
                                {
 | 
			
		||||
                                    LogMessage.LogWarning(ex, ToString());
 | 
			
		||||
                                }
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
                            if (isSuccess)
 | 
			
		||||
                                producer.Flush(cancellationToken);
 | 
			
		||||
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    catch (Exception ex)
 | 
			
		||||
                    {
 | 
			
		||||
                        LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (!driverPropertys.IsInterval)
 | 
			
		||||
            {
 | 
			
		||||
                ////变化推送
 | 
			
		||||
                var devList = _collectDeviceRunTimes.ToListWithDequeue();
 | 
			
		||||
                if (devList?.Count != 0)
 | 
			
		||||
                {
 | 
			
		||||
                    //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
                    var devData = devList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                    foreach (var item in devData)
 | 
			
		||||
                    {
 | 
			
		||||
                        try
 | 
			
		||||
                        {
 | 
			
		||||
                            if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                            {
 | 
			
		||||
                                await KafKaUp(driverPropertys.DeviceTopic, item.ToJsonString(), cancellationToken);
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                break;
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        catch (Exception ex)
 | 
			
		||||
                        {
 | 
			
		||||
                            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    if (isSuccess)
 | 
			
		||||
                        producer.Flush(cancellationToken);
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                if (exDeviceTimerTick.IsTickHappen())
 | 
			
		||||
                {
 | 
			
		||||
                    var devList = _collectDevice.Adapt<List<DeviceData>>();
 | 
			
		||||
                    if (devList?.Count != 0)
 | 
			
		||||
                    {
 | 
			
		||||
                        //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
                        var devData = devList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                        foreach (var item in devData)
 | 
			
		||||
                        {
 | 
			
		||||
                            try
 | 
			
		||||
                            {
 | 
			
		||||
                                if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                                {
 | 
			
		||||
                                    await KafKaUp(driverPropertys.DeviceTopic, item.ToJsonString(), cancellationToken);
 | 
			
		||||
                                }
 | 
			
		||||
                                else
 | 
			
		||||
                                {
 | 
			
		||||
                                    break;
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                            catch (Exception ex)
 | 
			
		||||
                            {
 | 
			
		||||
                                LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        if (isSuccess)
 | 
			
		||||
                            producer.Flush(cancellationToken);
 | 
			
		||||
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (driverPropertys.CycleInterval > UploadDeviceThread.CycleInterval + 50)
 | 
			
		||||
        {
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                await Task.Delay(driverPropertys.CycleInterval - UploadDeviceThread.CycleInterval, cancellationToken);
 | 
			
		||||
            }
 | 
			
		||||
            catch
 | 
			
		||||
            {
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public override bool IsConnected() => isSuccess;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        _collectDevice.Where(a => _uploadVariables.Select(b => b.DeviceId).Contains(a.Id)).ForEach(a =>
 | 
			
		||||
        {
 | 
			
		||||
            a.DeviceStatusChange -= DeviceStatusChange;
 | 
			
		||||
        });
 | 
			
		||||
        _uploadVariables?.ForEach(a =>
 | 
			
		||||
        {
 | 
			
		||||
            a.VariableValueChange -= VariableValueChange;
 | 
			
		||||
        });
 | 
			
		||||
        producer?.Dispose();
 | 
			
		||||
        _uploadVariables = null;
 | 
			
		||||
        _collectDeviceRunTimes.Clear();
 | 
			
		||||
        _collectVariableRunTimes.Clear();
 | 
			
		||||
        base.Dispose(disposing);
 | 
			
		||||
    }
 | 
			
		||||
    private List<CollectDeviceRunTime> _collectDevice;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 初始化
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="device"></param>
 | 
			
		||||
    protected override void Init(UploadDeviceRunTime device)
 | 
			
		||||
    {
 | 
			
		||||
        #region Kafka 生产者
 | 
			
		||||
        //1、生产者配置
 | 
			
		||||
        producerconfig = new ProducerConfig
 | 
			
		||||
        {
 | 
			
		||||
            BootstrapServers = driverPropertys.BootStrapServers,
 | 
			
		||||
            ClientId = driverPropertys.ClientId,
 | 
			
		||||
        };
 | 
			
		||||
        //2、创建生产者
 | 
			
		||||
        producerBuilder = new ProducerBuilder<Null, string>(producerconfig);
 | 
			
		||||
        //3、错误日志监视
 | 
			
		||||
        producerBuilder.SetErrorHandler((p, msg) =>
 | 
			
		||||
        {
 | 
			
		||||
            isSuccess = false;
 | 
			
		||||
            LogMessage?.LogWarning(msg.Reason);
 | 
			
		||||
        });
 | 
			
		||||
        //kafka
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            producer = producerBuilder.Build();
 | 
			
		||||
        }
 | 
			
		||||
        catch (DllNotFoundException)
 | 
			
		||||
        {
 | 
			
		||||
            if (!Library.IsLoaded)
 | 
			
		||||
            {
 | 
			
		||||
                string fileEx = ".dll";
 | 
			
		||||
                string osStr = "win-";
 | 
			
		||||
                if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
 | 
			
		||||
                {
 | 
			
		||||
                    osStr = "win-";
 | 
			
		||||
                    fileEx = ".dll";
 | 
			
		||||
                }
 | 
			
		||||
                else if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux))
 | 
			
		||||
                {
 | 
			
		||||
                    osStr = "linux-";
 | 
			
		||||
                    fileEx = ".so";
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
 | 
			
		||||
                    osStr = "osx-";
 | 
			
		||||
                    fileEx = ".dylib";
 | 
			
		||||
                }
 | 
			
		||||
                osStr += RuntimeInformation.ProcessArchitecture.ToString().ToLower();
 | 
			
		||||
 | 
			
		||||
                var pathToLibrd = System.IO.Path.Combine(AppContext.BaseDirectory, "Plugins", "ThingsGateway.Plugin.Kafka", "runtimes", osStr, "native", $"librdkafka{fileEx}");
 | 
			
		||||
                Library.Load(pathToLibrd);
 | 
			
		||||
            }
 | 
			
		||||
            producer = producerBuilder.Build();
 | 
			
		||||
        }
 | 
			
		||||
        #endregion
 | 
			
		||||
 | 
			
		||||
        _globalDeviceData = App.GetService<GlobalDeviceData>();
 | 
			
		||||
 | 
			
		||||
        var tags = _globalDeviceData.AllVariables.Where(a => a.VariablePropertys.ContainsKey(device.Id))
 | 
			
		||||
           .Where(b => GetPropertyValue(b, nameof(variablePropertys.Enable)).GetBoolValue()).ToList();
 | 
			
		||||
 | 
			
		||||
        _uploadVariables = tags;
 | 
			
		||||
 | 
			
		||||
        _collectDevice = _globalDeviceData.CollectDevices.Where(a => _uploadVariables.Select(b => b.DeviceId).Contains(a.Id)).ToList();
 | 
			
		||||
 | 
			
		||||
        _collectDevice.ForEach(a =>
 | 
			
		||||
        {
 | 
			
		||||
            a.DeviceStatusChange += DeviceStatusChange;
 | 
			
		||||
        });
 | 
			
		||||
        _uploadVariables.ForEach(a =>
 | 
			
		||||
        {
 | 
			
		||||
            a.VariableValueChange += VariableValueChange;
 | 
			
		||||
        });
 | 
			
		||||
        if (driverPropertys.UploadInterval <= 1000) driverPropertys.UploadInterval = 1000;
 | 
			
		||||
        exVariableTimerTick = new(driverPropertys.UploadInterval);
 | 
			
		||||
        exDeviceTimerTick = new(driverPropertys.UploadInterval);
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void DeviceStatusChange(CollectDeviceRunTime collectDeviceRunTime)
 | 
			
		||||
    {
 | 
			
		||||
        if (driverPropertys?.IsInterval != true)
 | 
			
		||||
            _collectDeviceRunTimes.Enqueue(collectDeviceRunTime.Adapt<DeviceData>());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task KafKaUp(string topic, string payLoad, CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            using CancellationTokenSource cancellationTokenSource = new();
 | 
			
		||||
            using CancellationTokenSource stoppingToken = CancellationTokenSource.CreateLinkedTokenSource(cancellationTokenSource.Token, cancellationToken);
 | 
			
		||||
            Task<DeliveryResult<Null, string>> resultTask = producer.ProduceAsync(topic, new Message<Null, string> { Value = payLoad }, stoppingToken.Token);
 | 
			
		||||
            var timeOutResult = await Task.WhenAny(resultTask, Task.Delay(driverPropertys.TimeOut, stoppingToken.Token));
 | 
			
		||||
            if (timeOutResult == resultTask)
 | 
			
		||||
            {
 | 
			
		||||
                var result = (timeOutResult as Task<DeliveryResult<Null, string>>).Result;
 | 
			
		||||
                if (result.Status != PersistenceStatus.Persisted)
 | 
			
		||||
                {
 | 
			
		||||
                    isSuccess = false;
 | 
			
		||||
                    await CacheDb.AddCacheData(topic, payLoad, driverPropertys.CacheMaxCount);
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    isSuccess = true;
 | 
			
		||||
                    //连接成功时补发缓存数据
 | 
			
		||||
                    var cacheData = await CacheDb.GetCacheData();
 | 
			
		||||
                    foreach (var item in cacheData)
 | 
			
		||||
                    {
 | 
			
		||||
                        var cacheResult = await producer.ProduceAsync(item.Topic, new Message<Null, string> { Value = item.CacheStr }, stoppingToken.Token);
 | 
			
		||||
 | 
			
		||||
                        if (cacheResult.Status == PersistenceStatus.Persisted)
 | 
			
		||||
                        {
 | 
			
		||||
                            LogMessage.Trace($"{FoundationConst.LogMessageHeader}主题:{item.Topic}{Environment.NewLine}负载:{item.CacheStr}");
 | 
			
		||||
 | 
			
		||||
                            await CacheDb.DeleteCacheData(item.Id);
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    LogMessage.Trace($"{FoundationConst.LogMessageHeader}主题:{topic}{Environment.NewLine}负载:{payLoad}");
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                isSuccess = false;
 | 
			
		||||
                stoppingToken.Cancel();
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
            await CacheDb.AddCacheData(topic, payLoad, driverPropertys.CacheMaxCount);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    private void VariableValueChange(DeviceVariableRunTime collectVariableRunTime)
 | 
			
		||||
    {
 | 
			
		||||
        if (driverPropertys?.IsInterval != true)
 | 
			
		||||
            _collectVariableRunTimes.Enqueue(collectVariableRunTime.Adapt<VariableData>());
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,108 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Modbus;
 | 
			
		||||
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
public class ModbusRtu : CollectBase
 | 
			
		||||
{
 | 
			
		||||
    private readonly ModbusRtuProperty driverPropertys = new();
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override IReadWrite PLC => _plc;
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Modbus.ModbusRtu _plc;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => typeof(ModbusRtuDebugPage);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public override bool IsSupportRequest => true;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override IThingsGatewayBitConverter ThingsGatewayBitConverter { get => _plc?.ThingsGatewayBitConverter; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        await _plc?.ConnectAsync(cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        _plc?.SafeDispose();
 | 
			
		||||
        base.Dispose(disposing);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void InitDataAdapter()
 | 
			
		||||
    {
 | 
			
		||||
        _plc.SetDataAdapter();
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsConnected()
 | 
			
		||||
    {
 | 
			
		||||
        return _plc?.SerialSession?.CanSend == true;
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableSourceRead> LoadSourceRead(List<DeviceVariableRunTime> deviceVariables)
 | 
			
		||||
    {
 | 
			
		||||
        return _plc.LoadSourceRead<DeviceVariableSourceRead, DeviceVariableRunTime>(deviceVariables, driverPropertys.MaxPack);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
    {
 | 
			
		||||
        if (client == null)
 | 
			
		||||
        {
 | 
			
		||||
            FoundataionConfig.SetSerialProperty(new()
 | 
			
		||||
            {
 | 
			
		||||
                PortName = driverPropertys.PortName,
 | 
			
		||||
                BaudRate = driverPropertys.BaudRate,
 | 
			
		||||
                DataBits = driverPropertys.DataBits,
 | 
			
		||||
                Parity = driverPropertys.Parity,
 | 
			
		||||
                StopBits = driverPropertys.StopBits,
 | 
			
		||||
            })
 | 
			
		||||
                ;
 | 
			
		||||
            client = new SerialSession();
 | 
			
		||||
            ((SerialSession)client).Setup(FoundataionConfig);
 | 
			
		||||
        }
 | 
			
		||||
        //载入配置
 | 
			
		||||
        _plc = new((SerialSession)client)
 | 
			
		||||
        {
 | 
			
		||||
            Crc16CheckEnable = driverPropertys.Crc16CheckEnable,
 | 
			
		||||
            FrameTime = driverPropertys.FrameTime,
 | 
			
		||||
            CacheTimeout = driverPropertys.CacheTimeout,
 | 
			
		||||
            DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
            Station = driverPropertys.Station,
 | 
			
		||||
            TimeOut = driverPropertys.TimeOut
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override async Task<OperResult<byte[]>> ReadAsync(string address, int length, CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return await _plc.ReadAsync(address, length, cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,98 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Modbus;
 | 
			
		||||
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
public class ModbusRtuOverTcp : CollectBase
 | 
			
		||||
{
 | 
			
		||||
 | 
			
		||||
    private readonly ModbusRtuOverTcpProperty driverPropertys = new();
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override IReadWrite PLC => _plc;
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Modbus.ModbusRtuOverTcp _plc;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => typeof(ModbusRtuOverTcpDebugPage);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsSupportRequest => true;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override IThingsGatewayBitConverter ThingsGatewayBitConverter { get => _plc?.ThingsGatewayBitConverter; }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        await _plc?.ConnectAsync(cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void InitDataAdapter()
 | 
			
		||||
    {
 | 
			
		||||
        _plc.SetDataAdapter();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsConnected()
 | 
			
		||||
    {
 | 
			
		||||
        return _plc?.TcpClient?.CanSend == true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableSourceRead> LoadSourceRead(List<DeviceVariableRunTime> deviceVariables)
 | 
			
		||||
    {
 | 
			
		||||
        return _plc.LoadSourceRead<DeviceVariableSourceRead, DeviceVariableRunTime>(deviceVariables, driverPropertys.MaxPack);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        _plc?.SafeDispose();
 | 
			
		||||
        base.Dispose(disposing);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
    {
 | 
			
		||||
        if (client == null)
 | 
			
		||||
        {
 | 
			
		||||
            FoundataionConfig.SetRemoteIPHost(new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}"))
 | 
			
		||||
                ;
 | 
			
		||||
            client = new TcpClient();
 | 
			
		||||
            ((TcpClient)client).Setup(FoundataionConfig);
 | 
			
		||||
        }
 | 
			
		||||
        //载入配置
 | 
			
		||||
        _plc = new((TcpClient)client)
 | 
			
		||||
        {
 | 
			
		||||
            Crc16CheckEnable = driverPropertys.Crc16CheckEnable,
 | 
			
		||||
            FrameTime = driverPropertys.FrameTime,
 | 
			
		||||
            CacheTimeout = driverPropertys.CacheTimeout,
 | 
			
		||||
            DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
            ConnectTimeOut = driverPropertys.ConnectTimeOut,
 | 
			
		||||
            Station = driverPropertys.Station,
 | 
			
		||||
            TimeOut = driverPropertys.TimeOut
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override async Task<OperResult<byte[]>> ReadAsync(string address, int length, CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return await _plc.ReadAsync(address, length, cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,102 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Modbus;
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class ModbusRtuOverUdp : CollectBase
 | 
			
		||||
{
 | 
			
		||||
    private readonly ModbusRtuOverUdpProperty driverPropertys = new();
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override IReadWrite PLC => _plc;
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Modbus.ModbusRtuOverUdp _plc;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => typeof(ModbusRtuOverUdpDebugPage);
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsSupportRequest => true;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override IThingsGatewayBitConverter ThingsGatewayBitConverter { get => _plc?.ThingsGatewayBitConverter; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        _plc.Disconnect();
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return _plc.ConnectAsync(cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void InitDataAdapter()
 | 
			
		||||
    {
 | 
			
		||||
        _plc.SetDataAdapter();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsConnected()
 | 
			
		||||
    {
 | 
			
		||||
        return _plc?.UdpSession?.CanSend == true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableSourceRead> LoadSourceRead(List<DeviceVariableRunTime> deviceVariables)
 | 
			
		||||
    {
 | 
			
		||||
        return _plc.LoadSourceRead<DeviceVariableSourceRead, DeviceVariableRunTime>(deviceVariables, driverPropertys.MaxPack);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        _plc?.SafeDispose();
 | 
			
		||||
        base.Dispose(disposing);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
    {
 | 
			
		||||
        if (client == null)
 | 
			
		||||
        {
 | 
			
		||||
            FoundataionConfig.SetRemoteIPHost(new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}"))
 | 
			
		||||
                .SetBindIPHost(new IPHost(0))
 | 
			
		||||
                ;
 | 
			
		||||
 | 
			
		||||
            client = new UdpSession();
 | 
			
		||||
            ((UdpSession)client).Setup(FoundataionConfig);
 | 
			
		||||
        }
 | 
			
		||||
        //载入配置
 | 
			
		||||
        _plc = new((UdpSession)client)
 | 
			
		||||
        {
 | 
			
		||||
            Crc16CheckEnable = driverPropertys.Crc16CheckEnable,
 | 
			
		||||
            FrameTime = driverPropertys.FrameTime,
 | 
			
		||||
            DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
            Station = driverPropertys.Station,
 | 
			
		||||
            TimeOut = driverPropertys.TimeOut
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override async Task<OperResult<byte[]>> ReadAsync(string address, int length, CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return await _plc.ReadAsync(address, length, cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,241 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using Furion;
 | 
			
		||||
 | 
			
		||||
using Microsoft.Extensions.Logging;
 | 
			
		||||
 | 
			
		||||
using SqlSugar;
 | 
			
		||||
 | 
			
		||||
using System.Collections.Concurrent;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Adapter.Modbus;
 | 
			
		||||
using ThingsGateway.Foundation.Extension;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.ConcurrentQueue;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.String;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Modbus;
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class ModbusSerialServer : UpLoadBase
 | 
			
		||||
{
 | 
			
		||||
 | 
			
		||||
    private readonly ModbusSerialServerProperty driverPropertys = new();
 | 
			
		||||
    private readonly ModbusSerialServerVariableProperty variablePropertys = new();
 | 
			
		||||
    private Dictionary<ModbusAddress, DeviceVariableRunTime> _ModbusTags;
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Modbus.ModbusSerialServer _plc;
 | 
			
		||||
    private ConcurrentQueue<(string, DeviceVariableRunTime)> Values = new();
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => typeof(ModbusSerialServerDebugPage);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override UpDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableRunTime> UploadVariables => _ModbusTags?.Values.ToList();
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override VariablePropertyBase VariablePropertys => variablePropertys;
 | 
			
		||||
    RpcSingletonService RpcCore { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return _plc?.ConnectAsync(cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task ExecuteAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        var list = Values.ToListWithDequeue();
 | 
			
		||||
        foreach (var item in list)
 | 
			
		||||
        {
 | 
			
		||||
            if (cancellationToken.IsCancellationRequested)
 | 
			
		||||
                break;
 | 
			
		||||
            var type = GetPropertyValue(item.Item2, nameof(ModbusSerialServerVariableProperty.ModbusType));
 | 
			
		||||
            if (Enum.TryParse<DataTypeEnum>(type, out DataTypeEnum result))
 | 
			
		||||
            {
 | 
			
		||||
                await _plc.WriteAsync(item.Item1, item.Item2.Value?.ToString(), 1, result, cancellationToken);
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                await _plc.WriteAsync(item.Item1, item.Item2.Value?.ToString(), 1, item.Item2.DataTypeEnum, cancellationToken);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (driverPropertys.CycleInterval > UploadDeviceThread.CycleInterval + 50)
 | 
			
		||||
        {
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                await Task.Delay(driverPropertys.CycleInterval - UploadDeviceThread.CycleInterval, cancellationToken);
 | 
			
		||||
            }
 | 
			
		||||
            catch
 | 
			
		||||
            {
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsConnected()
 | 
			
		||||
    {
 | 
			
		||||
        return _plc?.SerialSession?.CanSend == true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        _ModbusTags?.Values?.ToList()?.ForEach(a => a.VariableValueChange -= VariableValueChange);
 | 
			
		||||
        if (_plc != null)
 | 
			
		||||
            _plc.WriteData -= WriteData;
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        _plc?.SafeDispose();
 | 
			
		||||
        _ModbusTags?.Clear();
 | 
			
		||||
        _ModbusTags = null;
 | 
			
		||||
        Values?.Clear();
 | 
			
		||||
        Values = null;
 | 
			
		||||
        base.Dispose(disposing);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(UploadDeviceRunTime device)
 | 
			
		||||
    {
 | 
			
		||||
        var service = new SerialSession();
 | 
			
		||||
        FoundataionConfig.SetSerialProperty(new()
 | 
			
		||||
        {
 | 
			
		||||
            PortName = driverPropertys.PortName,
 | 
			
		||||
            BaudRate = driverPropertys.BaudRate,
 | 
			
		||||
            DataBits = driverPropertys.DataBits,
 | 
			
		||||
            Parity = driverPropertys.Parity,
 | 
			
		||||
            StopBits = driverPropertys.StopBits,
 | 
			
		||||
        })
 | 
			
		||||
            ;
 | 
			
		||||
        service = new SerialSession();
 | 
			
		||||
        ((SerialSession)service).Setup(FoundataionConfig);
 | 
			
		||||
        //载入配置
 | 
			
		||||
        _plc = new(service)
 | 
			
		||||
        {
 | 
			
		||||
            DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
            Station = driverPropertys.Station,
 | 
			
		||||
            CacheTimeout = driverPropertys.CacheTimeout,
 | 
			
		||||
            MulStation = driverPropertys.MulStation
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
        var _globalDeviceData = App.GetService<GlobalDeviceData>();
 | 
			
		||||
        var tags = _globalDeviceData.AllVariables.Where(a => a.VariablePropertys.ContainsKey(device.Id))
 | 
			
		||||
            .Where(b => !string.IsNullOrEmpty(GetPropertyValue(b, nameof(variablePropertys.ServiceAddress))))
 | 
			
		||||
            .ToList();
 | 
			
		||||
 | 
			
		||||
        tags.ForEach(a =>
 | 
			
		||||
        {
 | 
			
		||||
            a.VariableValueChange += VariableValueChange;
 | 
			
		||||
            VariableValueChange(a);
 | 
			
		||||
        });
 | 
			
		||||
        _plc.WriteData += WriteData;
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            _ModbusTags = tags.ToDictionary(a =>
 | 
			
		||||
            {
 | 
			
		||||
                ModbusAddress address = ModbusAddress.ParseFrom(
 | 
			
		||||
                    GetPropertyValue(a, nameof(variablePropertys.ServiceAddress))
 | 
			
		||||
                    , driverPropertys.Station);
 | 
			
		||||
                return address;
 | 
			
		||||
            });
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
            tags.ForEach(a =>
 | 
			
		||||
            {
 | 
			
		||||
                a.VariableValueChange -= VariableValueChange;
 | 
			
		||||
            });
 | 
			
		||||
            _plc.WriteData -= WriteData;
 | 
			
		||||
        }
 | 
			
		||||
        RpcCore = App.GetService<RpcSingletonService>();
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    private void VariableValueChange(DeviceVariableRunTime collectVariableRunTime)
 | 
			
		||||
    {
 | 
			
		||||
        var address = GetPropertyValue(collectVariableRunTime, nameof(variablePropertys.ServiceAddress));
 | 
			
		||||
        if (address != null && collectVariableRunTime.Value != null)
 | 
			
		||||
        {
 | 
			
		||||
            Values?.Enqueue((address, collectVariableRunTime));
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// RPC写入
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="address"></param>
 | 
			
		||||
    /// <param name="bytes"></param>
 | 
			
		||||
    /// <param name="thingsGatewayBitConverter"></param>
 | 
			
		||||
    /// <param name="client"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    private async Task<OperResult> WriteData(ModbusAddress address, byte[] bytes, IThingsGatewayBitConverter thingsGatewayBitConverter, SerialSession client)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var tag = _ModbusTags.FirstOrDefault(a => a.Key?.AddressStart == address.AddressStart && a.Key?.Station == address.Station && a.Key?.ReadFunction == address.ReadFunction);
 | 
			
		||||
 | 
			
		||||
            if (tag.Value == null) return OperResult.CreateSuccessResult();
 | 
			
		||||
            var enable =
 | 
			
		||||
                GetPropertyValue(tag.Value, nameof(variablePropertys.VariableRpcEnable)).ToBoolean()
 | 
			
		||||
                && driverPropertys.DeviceRpcEnable;
 | 
			
		||||
            if (!enable) return new OperResult("不允许写入");
 | 
			
		||||
            var type = GetPropertyValue(tag.Value, nameof(ModbusSerialServerVariableProperty.ModbusType));
 | 
			
		||||
            var addressStr = GetPropertyValue(tag.Value, nameof(ModbusSerialServerVariableProperty.ServiceAddress));
 | 
			
		||||
            if (Enum.TryParse<DataTypeEnum>(type, out DataTypeEnum result))
 | 
			
		||||
            {
 | 
			
		||||
                var result1 = await RpcCore.InvokeDeviceMethodAsync($"{nameof(ModbusSerialServer)}-{CurrentDevice.Name}",
 | 
			
		||||
               new Dictionary<string, string>
 | 
			
		||||
{
 | 
			
		||||
    {
 | 
			
		||||
                       tag.Value.Name,
 | 
			
		||||
                       thingsGatewayBitConverter.GetDataFormBytes(addressStr ,  bytes,result).ToString()
 | 
			
		||||
 | 
			
		||||
                   },
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
                );
 | 
			
		||||
                return result1.FirstOrDefault().Value;
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                var result1 = await RpcCore.InvokeDeviceMethodAsync($"{nameof(ModbusSerialServer)}-{CurrentDevice.Name}",
 | 
			
		||||
               new Dictionary<string, string>
 | 
			
		||||
{
 | 
			
		||||
    {
 | 
			
		||||
                       tag.Value.Name,
 | 
			
		||||
                       thingsGatewayBitConverter.GetDataFormBytes(addressStr ,   bytes,result).ToString()
 | 
			
		||||
                   },
 | 
			
		||||
}
 | 
			
		||||
                );
 | 
			
		||||
                return result1.FirstOrDefault().Value;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult(ex.Message);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,102 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Modbus;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class ModbusTcp : CollectBase
 | 
			
		||||
{
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override IReadWrite PLC => _plc;
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Modbus.ModbusTcp _plc;
 | 
			
		||||
 | 
			
		||||
    private readonly ModbusTcpProperty driverPropertys = new();
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => typeof(ModbusTcpDebugPage);
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsSupportRequest => true;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override IThingsGatewayBitConverter ThingsGatewayBitConverter { get => _plc?.ThingsGatewayBitConverter; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        await _plc?.ConnectAsync(cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void InitDataAdapter()
 | 
			
		||||
    {
 | 
			
		||||
        _plc.SetDataAdapter();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsConnected()
 | 
			
		||||
    {
 | 
			
		||||
        return _plc?.TcpClient?.CanSend == true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableSourceRead> LoadSourceRead(List<DeviceVariableRunTime> deviceVariables)
 | 
			
		||||
    {
 | 
			
		||||
        return _plc.LoadSourceRead<DeviceVariableSourceRead, DeviceVariableRunTime>(deviceVariables, driverPropertys.MaxPack);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        _plc?.SafeDispose();
 | 
			
		||||
        base.Dispose(disposing);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
    {
 | 
			
		||||
        if (client == null)
 | 
			
		||||
        {
 | 
			
		||||
            FoundataionConfig.SetRemoteIPHost(new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}"))
 | 
			
		||||
                ;
 | 
			
		||||
            client = new TcpClient();
 | 
			
		||||
            ((TcpClient)client).Setup(FoundataionConfig);
 | 
			
		||||
        }
 | 
			
		||||
        //载入配置
 | 
			
		||||
        _plc = new((TcpClient)client)
 | 
			
		||||
        {
 | 
			
		||||
            DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
            FrameTime = driverPropertys.FrameTime,
 | 
			
		||||
            CacheTimeout = driverPropertys.CacheTimeout,
 | 
			
		||||
            ConnectTimeOut = driverPropertys.ConnectTimeOut,
 | 
			
		||||
            Station = driverPropertys.Station,
 | 
			
		||||
            TimeOut = driverPropertys.TimeOut,
 | 
			
		||||
            IsCheckMessageId = driverPropertys.MessageIdCheckEnable
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override async Task<OperResult<byte[]>> ReadAsync(string address, int length, CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return await _plc.ReadAsync(address, length, cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,105 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Modbus;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class ModbusTcpDtu : CollectBase
 | 
			
		||||
{
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override IReadWrite PLC => _plc;
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Modbus.ModbusTcpDtu _plc;
 | 
			
		||||
 | 
			
		||||
    private readonly ModbusTcpProperty driverPropertys = new();
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => typeof(ModbusTcpDebugPage);
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsSupportRequest => true;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override IThingsGatewayBitConverter ThingsGatewayBitConverter { get => _plc?.ThingsGatewayBitConverter; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        await _plc?.ConnectAsync(cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void InitDataAdapter()
 | 
			
		||||
    {
 | 
			
		||||
        _plc.SetDataAdapter();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsConnected()
 | 
			
		||||
    {
 | 
			
		||||
        return _plc?.TcpService?.ServerState == ServerState.Running;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableSourceRead> LoadSourceRead(List<DeviceVariableRunTime> deviceVariables)
 | 
			
		||||
    {
 | 
			
		||||
        return _plc.LoadSourceRead<DeviceVariableSourceRead, DeviceVariableRunTime>(deviceVariables, driverPropertys.MaxPack);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        _plc?.SafeDispose();
 | 
			
		||||
        base.Dispose(disposing);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
    {
 | 
			
		||||
        if (client == null)
 | 
			
		||||
        {
 | 
			
		||||
            IPHost iPHost = new(driverPropertys.Port);
 | 
			
		||||
            if (!string.IsNullOrEmpty(driverPropertys.IP))
 | 
			
		||||
            {
 | 
			
		||||
                iPHost = new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}");
 | 
			
		||||
            }
 | 
			
		||||
            FoundataionConfig.SetListenIPHosts(new IPHost[] { iPHost });
 | 
			
		||||
            client = new TcpService();
 | 
			
		||||
            ((TcpService)client).Setup(FoundataionConfig);
 | 
			
		||||
        }
 | 
			
		||||
        //载入配置
 | 
			
		||||
        _plc = new((TcpService)client)
 | 
			
		||||
        {
 | 
			
		||||
            DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
            FrameTime = driverPropertys.FrameTime,
 | 
			
		||||
            CacheTimeout = driverPropertys.CacheTimeout,
 | 
			
		||||
            ConnectTimeOut = driverPropertys.ConnectTimeOut,
 | 
			
		||||
            Station = driverPropertys.Station,
 | 
			
		||||
            TimeOut = driverPropertys.TimeOut,
 | 
			
		||||
            IsCheckMessageId = driverPropertys.MessageIdCheckEnable
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override async Task<OperResult<byte[]>> ReadAsync(string address, int length, CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return await _plc.ReadAsync(address, length, cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,237 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using Furion;
 | 
			
		||||
 | 
			
		||||
using Microsoft.Extensions.Logging;
 | 
			
		||||
 | 
			
		||||
using SqlSugar;
 | 
			
		||||
 | 
			
		||||
using System.Collections.Concurrent;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Adapter.Modbus;
 | 
			
		||||
using ThingsGateway.Foundation.Extension;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.ConcurrentQueue;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.String;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Modbus;
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class ModbusTcpServer : UpLoadBase
 | 
			
		||||
{
 | 
			
		||||
 | 
			
		||||
    private readonly ModbusTcpServerProperty driverPropertys = new();
 | 
			
		||||
    private readonly ModbusTcpServerVariableProperty variablePropertys = new();
 | 
			
		||||
    private Dictionary<ModbusAddress, DeviceVariableRunTime> _ModbusTags;
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Modbus.ModbusTcpServer _plc;
 | 
			
		||||
    private ConcurrentQueue<(string, DeviceVariableRunTime)> Values = new();
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => typeof(ModbusTcpServerDebugPage);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override UpDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableRunTime> UploadVariables => _ModbusTags?.Values.ToList();
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override VariablePropertyBase VariablePropertys => variablePropertys;
 | 
			
		||||
    RpcSingletonService RpcCore { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return _plc?.ConnectAsync(cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task ExecuteAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        var list = Values.ToListWithDequeue();
 | 
			
		||||
        foreach (var item in list)
 | 
			
		||||
        {
 | 
			
		||||
            if (cancellationToken.IsCancellationRequested)
 | 
			
		||||
                break;
 | 
			
		||||
            var type = GetPropertyValue(item.Item2, nameof(ModbusTcpServerVariableProperty.ModbusType));
 | 
			
		||||
            if (Enum.TryParse<DataTypeEnum>(type, out DataTypeEnum result))
 | 
			
		||||
            {
 | 
			
		||||
                await _plc.WriteAsync(item.Item1, item.Item2.Value?.ToString(), 1, result, cancellationToken);
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                await _plc.WriteAsync(item.Item1, item.Item2.Value?.ToString(), 1, item.Item2.DataTypeEnum, cancellationToken);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (driverPropertys.CycleInterval > UploadDeviceThread.CycleInterval + 50)
 | 
			
		||||
        {
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                await Task.Delay(driverPropertys.CycleInterval - UploadDeviceThread.CycleInterval, cancellationToken);
 | 
			
		||||
            }
 | 
			
		||||
            catch
 | 
			
		||||
            {
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsConnected()
 | 
			
		||||
    {
 | 
			
		||||
        return _plc?.TcpService?.ServerState == ServerState.Running;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        _ModbusTags?.Values?.ToList()?.ForEach(a => a.VariableValueChange -= VariableValueChange);
 | 
			
		||||
        if (_plc != null)
 | 
			
		||||
            _plc.WriteData -= WriteData;
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        _plc?.SafeDispose();
 | 
			
		||||
        _ModbusTags?.Clear();
 | 
			
		||||
        _ModbusTags = null;
 | 
			
		||||
        Values?.Clear();
 | 
			
		||||
        Values = null;
 | 
			
		||||
        base.Dispose(disposing);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(UploadDeviceRunTime device)
 | 
			
		||||
    {
 | 
			
		||||
        IPHost iPHost = new(driverPropertys.Port);
 | 
			
		||||
        if (!string.IsNullOrEmpty(driverPropertys.IP))
 | 
			
		||||
        {
 | 
			
		||||
            iPHost = new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}");
 | 
			
		||||
        }
 | 
			
		||||
        FoundataionConfig.SetListenIPHosts(new IPHost[] { iPHost });
 | 
			
		||||
        var service = new TcpService();
 | 
			
		||||
        service.Setup(FoundataionConfig);
 | 
			
		||||
        //载入配置
 | 
			
		||||
        _plc = new(service)
 | 
			
		||||
        {
 | 
			
		||||
            DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
            Station = driverPropertys.Station,
 | 
			
		||||
            CacheTimeout = driverPropertys.CacheTimeout,
 | 
			
		||||
            MulStation = driverPropertys.MulStation
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
        var _globalDeviceData = App.GetService<GlobalDeviceData>();
 | 
			
		||||
        var tags = _globalDeviceData.AllVariables.Where(a => a.VariablePropertys.ContainsKey(device.Id))
 | 
			
		||||
            .Where(b => !string.IsNullOrEmpty(GetPropertyValue(b, nameof(variablePropertys.ServiceAddress))))
 | 
			
		||||
            .ToList();
 | 
			
		||||
 | 
			
		||||
        tags.ForEach(a =>
 | 
			
		||||
        {
 | 
			
		||||
            a.VariableValueChange += VariableValueChange;
 | 
			
		||||
            VariableValueChange(a);
 | 
			
		||||
        });
 | 
			
		||||
        _plc.WriteData += WriteData;
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            _ModbusTags = tags.ToDictionary(a =>
 | 
			
		||||
            {
 | 
			
		||||
                ModbusAddress address = ModbusAddress.ParseFrom(
 | 
			
		||||
                    GetPropertyValue(a, nameof(variablePropertys.ServiceAddress))
 | 
			
		||||
                    , driverPropertys.Station);
 | 
			
		||||
                return address;
 | 
			
		||||
            });
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
            tags.ForEach(a =>
 | 
			
		||||
            {
 | 
			
		||||
                a.VariableValueChange -= VariableValueChange;
 | 
			
		||||
            });
 | 
			
		||||
            _plc.WriteData -= WriteData;
 | 
			
		||||
        }
 | 
			
		||||
        RpcCore = App.GetService<RpcSingletonService>();
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    private void VariableValueChange(DeviceVariableRunTime collectVariableRunTime)
 | 
			
		||||
    {
 | 
			
		||||
        var address = GetPropertyValue(collectVariableRunTime, nameof(variablePropertys.ServiceAddress));
 | 
			
		||||
        if (address != null && collectVariableRunTime.Value != null)
 | 
			
		||||
        {
 | 
			
		||||
            Values?.Enqueue((address, collectVariableRunTime));
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// RPC写入
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="address"></param>
 | 
			
		||||
    /// <param name="bytes"></param>
 | 
			
		||||
    /// <param name="thingsGatewayBitConverter"></param>
 | 
			
		||||
    /// <param name="client"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    private async Task<OperResult> WriteData(ModbusAddress address, byte[] bytes, IThingsGatewayBitConverter thingsGatewayBitConverter, SocketClient client)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var tag = _ModbusTags.FirstOrDefault(a => a.Key?.AddressStart == address.AddressStart && a.Key?.Station == address.Station && a.Key?.ReadFunction == address.ReadFunction);
 | 
			
		||||
 | 
			
		||||
            if (tag.Value == null) return OperResult.CreateSuccessResult();
 | 
			
		||||
            var enable =
 | 
			
		||||
                GetPropertyValue(tag.Value, nameof(variablePropertys.VariableRpcEnable)).ToBoolean()
 | 
			
		||||
                && driverPropertys.DeviceRpcEnable;
 | 
			
		||||
            if (!enable) return new OperResult("不允许写入");
 | 
			
		||||
            var type = GetPropertyValue(tag.Value, nameof(ModbusTcpServerVariableProperty.ModbusType));
 | 
			
		||||
            var addressStr = GetPropertyValue(tag.Value, nameof(ModbusTcpServerVariableProperty.ServiceAddress));
 | 
			
		||||
            if (Enum.TryParse<DataTypeEnum>(type, out DataTypeEnum result))
 | 
			
		||||
            {
 | 
			
		||||
                var result1 = await RpcCore.InvokeDeviceMethodAsync($"{nameof(ModbusTcpServer)}-{CurrentDevice.Name}-{client.IP + ":" + client.Port}",
 | 
			
		||||
               new Dictionary<string, string>
 | 
			
		||||
{
 | 
			
		||||
    {
 | 
			
		||||
                       tag.Value.Name,
 | 
			
		||||
                       thingsGatewayBitConverter.GetDataFormBytes(addressStr ,  bytes,result).ToString()
 | 
			
		||||
 | 
			
		||||
                   },
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
                );
 | 
			
		||||
                return result1.FirstOrDefault().Value;
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                var result1 = await RpcCore.InvokeDeviceMethodAsync($"{nameof(ModbusTcpServer)}-{CurrentDevice.Name}-{client.IP + ":" + client.Port}",
 | 
			
		||||
               new Dictionary<string, string>
 | 
			
		||||
{
 | 
			
		||||
    {
 | 
			
		||||
                       tag.Value.Name,
 | 
			
		||||
                       thingsGatewayBitConverter.GetDataFormBytes(addressStr ,   bytes,result).ToString()
 | 
			
		||||
                   },
 | 
			
		||||
}
 | 
			
		||||
                );
 | 
			
		||||
                return result1.FirstOrDefault().Value;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            return new OperResult(ex.Message);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,101 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Modbus;
 | 
			
		||||
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
public class ModbusUdp : CollectBase
 | 
			
		||||
{
 | 
			
		||||
    private readonly ModbusUdpProperty driverPropertys = new();
 | 
			
		||||
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Modbus.ModbusUdp _plc;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => typeof(ModbusUdpDebugPage);
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsSupportRequest => true;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override IThingsGatewayBitConverter ThingsGatewayBitConverter { get => _plc?.ThingsGatewayBitConverter; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override IReadWrite PLC => _plc;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        _plc.Disconnect();
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return _plc.ConnectAsync(cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override void InitDataAdapter()
 | 
			
		||||
    {
 | 
			
		||||
        _plc.SetDataAdapter();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsConnected()
 | 
			
		||||
    {
 | 
			
		||||
        return _plc?.UdpSession?.CanSend == true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableSourceRead> LoadSourceRead(List<DeviceVariableRunTime> deviceVariables)
 | 
			
		||||
    {
 | 
			
		||||
        return _plc.LoadSourceRead<DeviceVariableSourceRead, DeviceVariableRunTime>(deviceVariables, driverPropertys.MaxPack);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        _plc?.Disconnect();
 | 
			
		||||
        _plc?.SafeDispose();
 | 
			
		||||
        base.Dispose(disposing);
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
    {
 | 
			
		||||
        if (client == null)
 | 
			
		||||
        {
 | 
			
		||||
            FoundataionConfig.SetRemoteIPHost(new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}"))
 | 
			
		||||
                .SetBindIPHost(new IPHost(0))
 | 
			
		||||
                ;
 | 
			
		||||
            client = new UdpSession();
 | 
			
		||||
            ((UdpSession)client).Setup(FoundataionConfig);
 | 
			
		||||
        }
 | 
			
		||||
        //载入配置
 | 
			
		||||
        _plc = new((UdpSession)client)
 | 
			
		||||
        {
 | 
			
		||||
            DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
            FrameTime = driverPropertys.FrameTime,
 | 
			
		||||
            Station = driverPropertys.Station,
 | 
			
		||||
            TimeOut = driverPropertys.TimeOut,
 | 
			
		||||
            IsCheckMessageId = driverPropertys.MessageIdCheckEnable
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override async Task<OperResult<byte[]>> ReadAsync(string address, int length, CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return await _plc.ReadAsync(address, length, cancellationToken);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,59 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@page "/ModbusRtu"
 | 
			
		||||
 | 
			
		||||
@using BlazorComponent;
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web;
 | 
			
		||||
@using Microsoft.JSInterop;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using Masa.Blazor;
 | 
			
		||||
@namespace ThingsGateway.Foundation.Demo
 | 
			
		||||
 | 
			
		||||
<SerialSessionPage @ref=SerialSessionPage></SerialSessionPage>
 | 
			
		||||
 | 
			
		||||
<MCard Elevation="1" Rounded="false" Class="pa-2" Style="width:100%">
 | 
			
		||||
    <div class="mb-4">驱动配置</div>
 | 
			
		||||
    <MRow Justify="JustifyTypes.Start" Align="AlignTypes.Center">
 | 
			
		||||
        @if (_plc != null)
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.FrameTime)) Dense HideDetails="@("auto")" @bind-Value=@_plc.FrameTime></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.CacheTimeout)) Dense HideDetails="@("auto")" @bind-Value=@_plc.CacheTimeout></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Station)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Station></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.TimeOut)) Dense HideDetails="@("auto")" @bind-Value=@_plc.TimeOut></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MSelect Class="ma-1" Outlined Style="max-width:200px" @bind-Value="_plc.DataFormat" Label="@(_plc.DescriptionWithOutSugar(x => x.DataFormat))"
 | 
			
		||||
                     Items=@(typeof(DataFormat).GetEnumListWithOutSugar())
 | 
			
		||||
                     MenuProps="@(props => { props.Auto = true; props.OffsetY = true; })"
 | 
			
		||||
                     ItemText=@((u) =>u.Description)
 | 
			
		||||
                     ItemValue=@(u =>(DataFormat)u.Value)
 | 
			
		||||
                     HideDetails=@("auto") Height="30"
 | 
			
		||||
                  Dense>
 | 
			
		||||
        </MSelect>
 | 
			
		||||
 | 
			
		||||
            <MCheckbox Class="ma-1" Style="max-width:200px" Label=@(_plc.DescriptionWithOutSugar(x => x.Crc16CheckEnable)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Crc16CheckEnable></MCheckbox>
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
    </MRow>
 | 
			
		||||
</MCard>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
<DriverDebugUIPage @ref=driverDebugUIPage Sections="_sections">
 | 
			
		||||
 | 
			
		||||
</DriverDebugUIPage>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@code {
 | 
			
		||||
    private readonly List<(string Code, string Language)> _sections = new();
 | 
			
		||||
}
 | 
			
		||||
@@ -1,40 +0,0 @@
 | 
			
		||||
<Project Sdk="Microsoft.NET.Sdk.Razor">
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	<Target Name="PostBuild" AfterTargets="PostBuildEvent">
 | 
			
		||||
 | 
			
		||||
		<Exec Command=" set dir="$(SolutionDir)Web\ThingsGateway.Web.Entry\bin\$(Configuration)\$(TargetFramework)\Plugins\$(AssemblyName)"
 if not exist %25dir%25  md %25dir%25  
copy "$(TargetDir)*Modbus*.dll"  %25dir%25


" />
 | 
			
		||||
 | 
			
		||||
	</Target>
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\DriverDebugUIBase.cs" Link="DebugPage\DriverDebugUIBase.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\DriverDebugUIPage.razor.cs" Link="DebugPage\DriverDebugUIPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\SerialSessionPage.razor.cs" Link="DebugPage\SerialSessionPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpClientPage.razor.cs" Link="DebugPage\TcpClientPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpServerPage.razor.cs" Link="DebugPage\TcpServerPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\UdpSessionPage.razor.cs" Link="DebugPage\UdpSessionPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\DriverDebugUIPage.razor" Link="DebugPage\DriverDebugUIPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\SerialSessionPage.razor" Link="DebugPage\SerialSessionPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpClientPage.razor" Link="DebugPage\TcpClientPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpServerPage.razor" Link="DebugPage\TcpServerPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\UdpSessionPage.razor" Link="DebugPage\UdpSessionPage.razor" />
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
		<ProjectReference Include="..\..\Foundation\ThingsGateway.Foundation.Adapter.Modbus\ThingsGateway.Foundation.Adapter.Modbus.csproj" />
 | 
			
		||||
		<ProjectReference Include="..\..\Web\ThingsGateway.Components\ThingsGateway.Components.csproj">
 | 
			
		||||
			<Private>false</Private>
 | 
			
		||||
			<CopyLocalSatelliteAssemblies>false</CopyLocalSatelliteAssemblies>
 | 
			
		||||
			<ExcludeAssets>runtime</ExcludeAssets>
 | 
			
		||||
		</ProjectReference>
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
</Project>
 | 
			
		||||
@@ -1,601 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using Furion;
 | 
			
		||||
 | 
			
		||||
using IoTSharp.Data;
 | 
			
		||||
 | 
			
		||||
using Mapster;
 | 
			
		||||
 | 
			
		||||
using Microsoft.Extensions.Logging;
 | 
			
		||||
 | 
			
		||||
using MQTTnet;
 | 
			
		||||
using MQTTnet.Client;
 | 
			
		||||
 | 
			
		||||
using System.Collections.Concurrent;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Admin.Core;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.ConcurrentQueue;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.Generic;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.String;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Mqtt;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// 参考IotSharpClient.SDK.MQTT
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class IotSharpClient : UpLoadBase
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// rpcmethodname存疑,定为自定义方法,在ThingsGateway上写入变量的方法固定为"Write"
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private const string WriteMethod = "WRITE";
 | 
			
		||||
 | 
			
		||||
    private readonly IotSharpClientProperty driverPropertys = new();
 | 
			
		||||
    private readonly EasyLock easyLock = new();
 | 
			
		||||
    private readonly IotSharpClientVariableProperty variablePropertys = new();
 | 
			
		||||
    private ConcurrentQueue<DeviceData> _collectDeviceRunTimes = new();
 | 
			
		||||
    private ConcurrentQueue<VariableData> _collectVariableRunTimes = new();
 | 
			
		||||
    private GlobalDeviceData _globalDeviceData;
 | 
			
		||||
 | 
			
		||||
    private IMqttClient _mqttClient;
 | 
			
		||||
 | 
			
		||||
    private MqttClientOptions _mqttClientOptions;
 | 
			
		||||
 | 
			
		||||
    private MqttClientSubscribeOptions _mqttSubscribeOptions;
 | 
			
		||||
 | 
			
		||||
    private RpcSingletonService _rpcCore;
 | 
			
		||||
    private List<DeviceVariableRunTime> _uploadVariables = new();
 | 
			
		||||
    private CollectDeviceWorker collectDeviceHostService;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => null;
 | 
			
		||||
    private TimerTick exVariableTimerTick;
 | 
			
		||||
    private TimerTick exDeviceTimerTick;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override UpDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableRunTime> UploadVariables => _uploadVariables;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override VariablePropertyBase VariablePropertys => variablePropertys;
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        if (_mqttClient != null)
 | 
			
		||||
        {
 | 
			
		||||
            var result = await TryMqttClientAsync(cancellationToken);
 | 
			
		||||
            if (!result.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                LogMessage?.LogWarning($"{ToString()}-连接MqttServer失败:{result.Message}");
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task ExecuteAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (!driverPropertys.IsInterval)
 | 
			
		||||
            {
 | 
			
		||||
                ////变化推送
 | 
			
		||||
                var varList = _collectVariableRunTimes.ToListWithDequeue();
 | 
			
		||||
                if (varList?.Count != 0)
 | 
			
		||||
                {
 | 
			
		||||
                    //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
                    var varData = varList.GroupBy(a => a.DeviceName).ToList();
 | 
			
		||||
                    foreach (var item in varData)
 | 
			
		||||
                    {
 | 
			
		||||
                        try
 | 
			
		||||
                        {
 | 
			
		||||
                            Dictionary<string, object> nameValueDict = new();
 | 
			
		||||
                            foreach (var pair in item)
 | 
			
		||||
                            {
 | 
			
		||||
                                //只用最新的变量值
 | 
			
		||||
                                nameValueDict.AddOrUpdate(pair.Name, pair.Value);
 | 
			
		||||
                            }
 | 
			
		||||
                            if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                            {
 | 
			
		||||
                                await MqttUp($"devices/{item.Key}/telemetry", nameValueDict.ToJsonString(), cancellationToken);
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                break;
 | 
			
		||||
                            }
 | 
			
		||||
 | 
			
		||||
                        }
 | 
			
		||||
                        catch (Exception ex)
 | 
			
		||||
                        {
 | 
			
		||||
                            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                if (exVariableTimerTick.IsTickHappen())
 | 
			
		||||
                {
 | 
			
		||||
                    try
 | 
			
		||||
                    {
 | 
			
		||||
                        var varList = _uploadVariables.Adapt<List<VariableData>>();
 | 
			
		||||
                        if (varList?.Count != 0)
 | 
			
		||||
                        {
 | 
			
		||||
                            //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
                            var varData = varList.GroupBy(a => a.DeviceName).ToList();
 | 
			
		||||
                            foreach (var item in varData)
 | 
			
		||||
                            {
 | 
			
		||||
                                try
 | 
			
		||||
                                {
 | 
			
		||||
                                    Dictionary<string, object> nameValueDict = new();
 | 
			
		||||
                                    foreach (var pair in item)
 | 
			
		||||
                                    {
 | 
			
		||||
                                        //只用最新的变量值
 | 
			
		||||
                                        nameValueDict.AddOrUpdate(pair.Name, pair.Value);
 | 
			
		||||
                                    }
 | 
			
		||||
                                    if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                                    {
 | 
			
		||||
                                        await MqttUp($"devices/{item.Key}/telemetry", nameValueDict.ToJsonString(), cancellationToken);
 | 
			
		||||
                                    }
 | 
			
		||||
                                    else
 | 
			
		||||
                                    {
 | 
			
		||||
                                        break;
 | 
			
		||||
                                    }
 | 
			
		||||
 | 
			
		||||
                                }
 | 
			
		||||
                                catch (Exception ex)
 | 
			
		||||
                                {
 | 
			
		||||
                                    LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                                }
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    catch (Exception ex)
 | 
			
		||||
                    {
 | 
			
		||||
                        LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (!driverPropertys.IsInterval)
 | 
			
		||||
            {
 | 
			
		||||
                ////变化推送
 | 
			
		||||
                var devList = _collectDeviceRunTimes.ToListWithDequeue();
 | 
			
		||||
                if (devList?.Count != 0)
 | 
			
		||||
                {
 | 
			
		||||
                    //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
                    var devData = devList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                    foreach (var item in devData)
 | 
			
		||||
                    {
 | 
			
		||||
                        try
 | 
			
		||||
                        {
 | 
			
		||||
                            if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                            {
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                break;
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        catch (Exception ex)
 | 
			
		||||
                        {
 | 
			
		||||
                            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                if (exDeviceTimerTick.IsTickHappen())
 | 
			
		||||
                {
 | 
			
		||||
                    var devList = _collectDevice.Adapt<List<DeviceData>>();
 | 
			
		||||
                    if (devList?.Count != 0)
 | 
			
		||||
                    {
 | 
			
		||||
                        //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
                        var devData = devList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                        foreach (var item in devData)
 | 
			
		||||
                        {
 | 
			
		||||
                            try
 | 
			
		||||
                            {
 | 
			
		||||
                                if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                                {
 | 
			
		||||
 | 
			
		||||
                                }
 | 
			
		||||
                                else
 | 
			
		||||
                                {
 | 
			
		||||
                                    break;
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                            catch (Exception ex)
 | 
			
		||||
                            {
 | 
			
		||||
                                LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (driverPropertys.CycleInterval > UploadDeviceThread.CycleInterval + 50)
 | 
			
		||||
        {
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                await Task.Delay(driverPropertys.CycleInterval - UploadDeviceThread.CycleInterval, cancellationToken);
 | 
			
		||||
            }
 | 
			
		||||
            catch
 | 
			
		||||
            {
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsConnected() => _mqttClient?.IsConnected == true;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override string ToString()
 | 
			
		||||
    {
 | 
			
		||||
        return $" {nameof(IotSharpClient)}-IP:{driverPropertys.IP}-Port:{driverPropertys.Port}-Accesstoken:{driverPropertys.Accesstoken}";
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            _globalDeviceData?.AllVariables?.ForEach(a => a.VariableValueChange -= VariableValueChange);
 | 
			
		||||
 | 
			
		||||
            _collectDevice?.ForEach(a =>
 | 
			
		||||
            {
 | 
			
		||||
                a.DeviceStatusChange -= DeviceStatusChange;
 | 
			
		||||
            });
 | 
			
		||||
            _mqttClient?.SafeDispose();
 | 
			
		||||
            _mqttClient = null;
 | 
			
		||||
            _uploadVariables = null;
 | 
			
		||||
            _collectDeviceRunTimes.Clear();
 | 
			
		||||
            _collectVariableRunTimes.Clear();
 | 
			
		||||
            _collectDeviceRunTimes = null;
 | 
			
		||||
            _collectVariableRunTimes = null;
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogError(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    private List<CollectDeviceRunTime> _collectDevice;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(UploadDeviceRunTime device)
 | 
			
		||||
    {
 | 
			
		||||
        var mqttFactory = new MqttFactory(new PrivateLogger(LogMessage));
 | 
			
		||||
        _mqttClientOptions = mqttFactory.CreateClientOptionsBuilder()
 | 
			
		||||
           .WithClientId(Guid.NewGuid().ToString())
 | 
			
		||||
           .WithCredentials(driverPropertys.Accesstoken)//账密
 | 
			
		||||
           .WithTcpServer(driverPropertys.IP, driverPropertys.Port)//服务器
 | 
			
		||||
           .WithCleanSession(true)
 | 
			
		||||
           .WithKeepAlivePeriod(TimeSpan.FromSeconds(120.0))
 | 
			
		||||
           .Build();
 | 
			
		||||
        _mqttSubscribeOptions = mqttFactory.CreateSubscribeOptionsBuilder()
 | 
			
		||||
            .WithTopicFilter(
 | 
			
		||||
                f =>
 | 
			
		||||
                {
 | 
			
		||||
                    f.WithTopic($"devices/+/rpc/request/+/+");//RPC控制请求,需要订阅
 | 
			
		||||
                })
 | 
			
		||||
 | 
			
		||||
            .Build();
 | 
			
		||||
        _mqttClient = mqttFactory.CreateMqttClient();
 | 
			
		||||
        _mqttClient.ConnectedAsync += MqttClient_ConnectedAsync;
 | 
			
		||||
        _mqttClient.ApplicationMessageReceivedAsync += MqttClient_ApplicationMessageReceivedAsync;
 | 
			
		||||
        _globalDeviceData = App.GetService<GlobalDeviceData>();
 | 
			
		||||
        _rpcCore = App.GetService<RpcSingletonService>();
 | 
			
		||||
        collectDeviceHostService = BackgroundServiceUtil.GetBackgroundService<CollectDeviceWorker>();
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        var tags = _globalDeviceData.AllVariables.Where(a => a.VariablePropertys.ContainsKey(device.Id))
 | 
			
		||||
           .Where(b => GetPropertyValue(b, nameof(variablePropertys.Enable)).GetBoolValue())
 | 
			
		||||
           .ToList();
 | 
			
		||||
 | 
			
		||||
        _uploadVariables = tags;
 | 
			
		||||
 | 
			
		||||
        _collectDevice = _globalDeviceData.CollectDevices.Where(a => _uploadVariables.Select(b => b.DeviceId).Contains(a.Id)).ToList();
 | 
			
		||||
 | 
			
		||||
        _collectDevice?.ForEach(a =>
 | 
			
		||||
        {
 | 
			
		||||
            a.DeviceStatusChange += DeviceStatusChange;
 | 
			
		||||
        });
 | 
			
		||||
        _uploadVariables.ForEach(a =>
 | 
			
		||||
        {
 | 
			
		||||
            a.VariableValueChange += VariableValueChange;
 | 
			
		||||
        });
 | 
			
		||||
        if (driverPropertys.UploadInterval <= 1000) driverPropertys.UploadInterval = 1000;
 | 
			
		||||
        exVariableTimerTick = new(driverPropertys.UploadInterval);
 | 
			
		||||
        exDeviceTimerTick = new(driverPropertys.UploadInterval);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void DeviceStatusChange(CollectDeviceRunTime collectDeviceRunTime)
 | 
			
		||||
    {
 | 
			
		||||
        if (driverPropertys?.IsInterval != true)
 | 
			
		||||
            _collectDeviceRunTimes.Enqueue(collectDeviceRunTime.Adapt<DeviceData>());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task MqttClient_ApplicationMessageReceivedAsync(MqttApplicationMessageReceivedEventArgs e)
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
        if (e.ApplicationMessage.Topic.StartsWith($"devices/") && e.ApplicationMessage.Topic.Contains("/rpc/request/"))
 | 
			
		||||
        {
 | 
			
		||||
            var tps = e.ApplicationMessage.Topic.Split(new char[] { '/' }, StringSplitOptions.RemoveEmptyEntries);
 | 
			
		||||
            var rpcmethodname = tps[4];
 | 
			
		||||
            var rpcdevicename = tps[1];
 | 
			
		||||
            var rpcrequestid = tps[5];
 | 
			
		||||
            if (!string.IsNullOrEmpty(rpcmethodname) && !string.IsNullOrEmpty(rpcdevicename) && !string.IsNullOrEmpty(rpcrequestid))
 | 
			
		||||
            {
 | 
			
		||||
                var rpcResponse = new RpcResponse()
 | 
			
		||||
                {
 | 
			
		||||
                    DeviceId = rpcdevicename,
 | 
			
		||||
                    ResponseId = rpcrequestid,
 | 
			
		||||
                    Method = rpcmethodname,
 | 
			
		||||
                    Success = false,
 | 
			
		||||
                    Data = "参数为空"
 | 
			
		||||
                };
 | 
			
		||||
                await SendResponseAsync(rpcResponse);
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
            if (!driverPropertys.DeviceRpcEnable)
 | 
			
		||||
            {
 | 
			
		||||
                var rpcResponse = new RpcResponse()
 | 
			
		||||
                {
 | 
			
		||||
                    DeviceId = rpcdevicename,
 | 
			
		||||
                    ResponseId = rpcrequestid,
 | 
			
		||||
                    Method = rpcmethodname,
 | 
			
		||||
                    Success = false,
 | 
			
		||||
                    Data = "不允许写入"
 | 
			
		||||
                };
 | 
			
		||||
                await SendResponseAsync(rpcResponse);
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
            //rpcmethodname定为自定义方法,在ThingsGateway上写入变量的方法固定为"Write"
 | 
			
		||||
            if (rpcmethodname.ToUpper() != WriteMethod)
 | 
			
		||||
            {
 | 
			
		||||
                var rpcResponse = new RpcResponse()
 | 
			
		||||
                {
 | 
			
		||||
                    DeviceId = rpcdevicename,
 | 
			
		||||
                    ResponseId = rpcrequestid,
 | 
			
		||||
                    Method = rpcmethodname,
 | 
			
		||||
                    Success = false,
 | 
			
		||||
                    Data = "不支持的方法"
 | 
			
		||||
                };
 | 
			
		||||
                await SendResponseAsync(rpcResponse);
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                RpcResponse rpcResponse = new();
 | 
			
		||||
                var nameValue = e.ApplicationMessage.ConvertPayloadToString().FromJsonString<List<KeyValuePair<string, string>>>();
 | 
			
		||||
                Dictionary<string, OperResult> results = new();
 | 
			
		||||
                if (nameValue?.Count > 0)
 | 
			
		||||
                {
 | 
			
		||||
                    foreach (var item in nameValue)
 | 
			
		||||
                    {
 | 
			
		||||
                        var tag = _uploadVariables.FirstOrDefault(a => a.Name == item.Key);
 | 
			
		||||
                        if (tag != null)
 | 
			
		||||
                        {
 | 
			
		||||
                            var rpcEnable =
 | 
			
		||||
GetPropertyValue(tag, nameof(variablePropertys.VariableRpcEnable)).ToBoolean()
 | 
			
		||||
&& driverPropertys.DeviceRpcEnable;
 | 
			
		||||
                            if (rpcEnable == true)
 | 
			
		||||
                            {
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                results.Add(item.Key, new OperResult("权限不足,变量不支持写入"));
 | 
			
		||||
                            }
 | 
			
		||||
 | 
			
		||||
                        }
 | 
			
		||||
                        else
 | 
			
		||||
                        {
 | 
			
		||||
                            results.Add(item.Key, new OperResult("不存在该变量"));
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    var result = await _rpcCore.InvokeDeviceMethodAsync(ToString() + "-" + rpcrequestid, nameValue
 | 
			
		||||
                        .Where(a => !results.Any(b => b.Key == a.Key))
 | 
			
		||||
                        .ToDictionary(a => a.Key, a => a.Value));
 | 
			
		||||
 | 
			
		||||
                    results.AddRange(result);
 | 
			
		||||
                    rpcResponse = new()
 | 
			
		||||
                    {
 | 
			
		||||
                        DeviceId = rpcdevicename,
 | 
			
		||||
                        ResponseId = rpcrequestid,
 | 
			
		||||
                        Method = rpcmethodname,
 | 
			
		||||
                        Success = !results.Any(a => !a.Value.IsSuccess),
 | 
			
		||||
                        Data = results.ToJsonString()
 | 
			
		||||
                    };
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    rpcResponse = new()
 | 
			
		||||
                    {
 | 
			
		||||
                        DeviceId = rpcdevicename,
 | 
			
		||||
                        ResponseId = rpcrequestid,
 | 
			
		||||
                        Method = rpcmethodname,
 | 
			
		||||
                        Success = false,
 | 
			
		||||
                        Data = "消息体参数无法解析"
 | 
			
		||||
                    };
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                await SendResponseAsync(rpcResponse);
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        async Task SendResponseAsync(RpcResponse rpcResponse)
 | 
			
		||||
        {
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                var topic = $"devices/{rpcResponse.DeviceId}/rpc/response/{rpcResponse.Method}/{rpcResponse.ResponseId}";
 | 
			
		||||
 | 
			
		||||
                var variableMessage = new MqttApplicationMessageBuilder()
 | 
			
		||||
.WithTopic($"{topic}")
 | 
			
		||||
.WithPayload(rpcResponse.ToJsonString()).Build();
 | 
			
		||||
                var isConnect = await TryMqttClientAsync(CancellationToken.None);
 | 
			
		||||
                if (isConnect.IsSuccess)
 | 
			
		||||
                    await _mqttClient.PublishAsync(variableMessage);
 | 
			
		||||
            }
 | 
			
		||||
            catch
 | 
			
		||||
            {
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task MqttClient_ConnectedAsync(MqttClientConnectedEventArgs arg)
 | 
			
		||||
    {
 | 
			
		||||
        var subResult = await _mqttClient.SubscribeAsync(_mqttSubscribeOptions);
 | 
			
		||||
 | 
			
		||||
        if (subResult.Items.Any(a => a.ResultCode > (MqttClientSubscribeResultCode)10))
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(subResult.Items
 | 
			
		||||
                .Where(a => a.ResultCode > (MqttClientSubscribeResultCode)10)
 | 
			
		||||
                .Select(a => a.ToString()).ToJsonString());
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 上传mqtt内容,并进行离线缓存
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="topic"></param>
 | 
			
		||||
    /// <param name="payLoad"></param>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    private async Task MqttUp(string topic, string payLoad, CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        var variableMessage = new MqttApplicationMessageBuilder()
 | 
			
		||||
.WithTopic(topic)
 | 
			
		||||
.WithPayload(payLoad).Build();
 | 
			
		||||
        var isConnect = await TryMqttClientAsync(cancellationToken);
 | 
			
		||||
        if (isConnect.IsSuccess)
 | 
			
		||||
        {
 | 
			
		||||
            //连接成功时补发缓存数据
 | 
			
		||||
            var cacheData = await CacheDb.GetCacheData();
 | 
			
		||||
            foreach (var item in cacheData)
 | 
			
		||||
            {
 | 
			
		||||
                var cacheMessage = new MqttApplicationMessageBuilder()
 | 
			
		||||
.WithTopic(item.Topic)
 | 
			
		||||
.WithPayload(item.CacheStr).Build();
 | 
			
		||||
                var cacheResult = await _mqttClient.PublishAsync(cacheMessage, cancellationToken);
 | 
			
		||||
                if (cacheResult.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    await CacheDb.DeleteCacheData(item.Id);
 | 
			
		||||
                    LogMessage.Trace($"{FoundationConst.LogMessageHeader}主题:{item.Topic}{Environment.NewLine}负载:{item.CacheStr}");
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            var result = await _mqttClient.PublishAsync(variableMessage, cancellationToken);
 | 
			
		||||
            if (!result.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                await CacheDb.AddCacheData(topic, payLoad, driverPropertys.CacheMaxCount);
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                LogMessage.Trace($"{FoundationConst.LogMessageHeader}主题:{topic}{Environment.NewLine}负载:{payLoad}");
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            await CacheDb.AddCacheData(topic, payLoad, driverPropertys.CacheMaxCount);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task<OperResult> TryMqttClientAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        if (_mqttClient?.IsConnected == true)
 | 
			
		||||
            return OperResult.CreateSuccessResult();
 | 
			
		||||
        return await Cilent();
 | 
			
		||||
 | 
			
		||||
        async Task<OperResult> Cilent()
 | 
			
		||||
        {
 | 
			
		||||
            if (_mqttClient?.IsConnected == true)
 | 
			
		||||
                return OperResult.CreateSuccessResult();
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                await easyLock.WaitAsync();
 | 
			
		||||
                if (_mqttClient?.IsConnected == true)
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
                using var timeoutToken = new CancellationTokenSource(TimeSpan.FromMilliseconds(driverPropertys.ConnectTimeOut));
 | 
			
		||||
                using CancellationTokenSource StoppingToken = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken, timeoutToken.Token);
 | 
			
		||||
                if (_mqttClient?.IsConnected == true)
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
                if (_mqttClient == null)
 | 
			
		||||
                    return new OperResult("未初始化");
 | 
			
		||||
                var result = await _mqttClient?.ConnectAsync(_mqttClientOptions, StoppingToken.Token);
 | 
			
		||||
                if (result.ResultCode == MqttClientConnectResultCode.Success)
 | 
			
		||||
                {
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult(result.ReasonString);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            catch (Exception ex)
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(ex);
 | 
			
		||||
            }
 | 
			
		||||
            finally
 | 
			
		||||
            {
 | 
			
		||||
                easyLock.Release();
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    private void VariableValueChange(DeviceVariableRunTime collectVariableRunTime)
 | 
			
		||||
    {
 | 
			
		||||
        if (driverPropertys?.IsInterval != true)
 | 
			
		||||
            _collectVariableRunTimes.Enqueue(collectVariableRunTime.Adapt<VariableData>());
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,569 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using Furion;
 | 
			
		||||
 | 
			
		||||
using Mapster;
 | 
			
		||||
 | 
			
		||||
using Microsoft.Extensions.Logging;
 | 
			
		||||
 | 
			
		||||
using MQTTnet;
 | 
			
		||||
using MQTTnet.Client;
 | 
			
		||||
 | 
			
		||||
using System.Collections.Concurrent;
 | 
			
		||||
using System.Text;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Extension.ConcurrentQueue;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.Generic;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.String;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Mqtt;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// MqttClient
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class MqttClient : UpLoadBase
 | 
			
		||||
{
 | 
			
		||||
    private readonly MqttClientProperty driverPropertys = new();
 | 
			
		||||
    private readonly EasyLock easyLock = new();
 | 
			
		||||
    private readonly MqttClientVariableProperty variablePropertys = new();
 | 
			
		||||
    private List<CollectDeviceRunTime> _collectDevice;
 | 
			
		||||
    private ConcurrentQueue<DeviceData> _collectDeviceRunTimes = new();
 | 
			
		||||
    private ConcurrentQueue<VariableData> _collectVariableRunTimes = new();
 | 
			
		||||
    private GlobalDeviceData _globalDeviceData;
 | 
			
		||||
    private IMqttClient _mqttClient;
 | 
			
		||||
 | 
			
		||||
    private MqttClientOptions _mqttClientOptions;
 | 
			
		||||
 | 
			
		||||
    private MqttClientSubscribeOptions _mqttSubscribeOptions;
 | 
			
		||||
 | 
			
		||||
    private RpcSingletonService _rpcCore;
 | 
			
		||||
 | 
			
		||||
    private List<DeviceVariableRunTime> _uploadVariables = new();
 | 
			
		||||
 | 
			
		||||
    private CollectDeviceWorker collectDeviceHostService;
 | 
			
		||||
    private TimerTick exDeviceTimerTick;
 | 
			
		||||
 | 
			
		||||
    private TimerTick exVariableTimerTick;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => typeof(MqttClientDebugPage);
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override UpDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableRunTime> UploadVariables => _uploadVariables;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override VariablePropertyBase VariablePropertys => variablePropertys;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        if (_mqttClient != null)
 | 
			
		||||
        {
 | 
			
		||||
            var result = await TryMqttClientAsync(cancellationToken);
 | 
			
		||||
            if (!result.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                LogMessage?.LogWarning($"{ToString()}-连接MqttServer失败:{result.Message}");
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task ExecuteAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (!driverPropertys.IsInterval)
 | 
			
		||||
            {
 | 
			
		||||
                ////变化推送
 | 
			
		||||
                var varList = _collectVariableRunTimes.ToListWithDequeue();
 | 
			
		||||
                if (varList?.Count != 0)
 | 
			
		||||
                {
 | 
			
		||||
                    //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
                    var varData = varList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                    foreach (var item in varData)
 | 
			
		||||
                    {
 | 
			
		||||
                        try
 | 
			
		||||
                        {
 | 
			
		||||
                            if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                            {
 | 
			
		||||
                                await MqttUp($"{driverPropertys.VariableTopic}", item.GetSciptListValue(driverPropertys.BigTextScriptVariableModel), cancellationToken);
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                break;
 | 
			
		||||
                            }
 | 
			
		||||
 | 
			
		||||
                        }
 | 
			
		||||
                        catch (Exception ex)
 | 
			
		||||
                        {
 | 
			
		||||
                            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                if (exVariableTimerTick.IsTickHappen())
 | 
			
		||||
                {
 | 
			
		||||
                    try
 | 
			
		||||
                    {
 | 
			
		||||
                        var varList = _uploadVariables.Adapt<List<VariableData>>();
 | 
			
		||||
                        if (varList?.Count != 0)
 | 
			
		||||
                        {
 | 
			
		||||
                            //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
                            var varData = varList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                            foreach (var item in varData)
 | 
			
		||||
                            {
 | 
			
		||||
                                try
 | 
			
		||||
                                {
 | 
			
		||||
                                    if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                                    {
 | 
			
		||||
 | 
			
		||||
                                        await MqttUp($"{driverPropertys.VariableTopic}", item.GetSciptListValue(driverPropertys.BigTextScriptVariableModel), cancellationToken);
 | 
			
		||||
 | 
			
		||||
                                    }
 | 
			
		||||
                                    else
 | 
			
		||||
                                    {
 | 
			
		||||
                                        break;
 | 
			
		||||
                                    }
 | 
			
		||||
 | 
			
		||||
                                }
 | 
			
		||||
                                catch (Exception ex)
 | 
			
		||||
                                {
 | 
			
		||||
                                    LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                                }
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    catch (Exception ex)
 | 
			
		||||
                    {
 | 
			
		||||
                        LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (!driverPropertys.IsInterval)
 | 
			
		||||
            {
 | 
			
		||||
                ////变化推送
 | 
			
		||||
                var devList = _collectDeviceRunTimes.ToListWithDequeue();
 | 
			
		||||
                if (devList?.Count != 0)
 | 
			
		||||
                {
 | 
			
		||||
                    //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
                    var devData = devList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                    foreach (var item in devData)
 | 
			
		||||
                    {
 | 
			
		||||
                        try
 | 
			
		||||
                        {
 | 
			
		||||
                            if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                            {
 | 
			
		||||
                                await MqttUp($"{driverPropertys.DeviceTopic}", item.GetSciptListValue(driverPropertys.BigTextScriptDeviceModel), cancellationToken);
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                break;
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        catch (Exception ex)
 | 
			
		||||
                        {
 | 
			
		||||
                            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                if (exDeviceTimerTick.IsTickHappen())
 | 
			
		||||
                {
 | 
			
		||||
                    var devList = _collectDevice.Adapt<List<DeviceData>>();
 | 
			
		||||
                    if (devList?.Count != 0)
 | 
			
		||||
                    {
 | 
			
		||||
                        //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
                        var devData = devList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                        foreach (var item in devData)
 | 
			
		||||
                        {
 | 
			
		||||
                            try
 | 
			
		||||
                            {
 | 
			
		||||
                                if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                                {
 | 
			
		||||
                                    await MqttUp($"{driverPropertys.DeviceTopic}", item.GetSciptListValue(driverPropertys.BigTextScriptDeviceModel), cancellationToken);
 | 
			
		||||
                                }
 | 
			
		||||
                                else
 | 
			
		||||
                                {
 | 
			
		||||
                                    break;
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                            catch (Exception ex)
 | 
			
		||||
                            {
 | 
			
		||||
                                LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (driverPropertys.CycleInterval > UploadDeviceThread.CycleInterval + 50)
 | 
			
		||||
        {
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                await Task.Delay(driverPropertys.CycleInterval - UploadDeviceThread.CycleInterval, cancellationToken);
 | 
			
		||||
            }
 | 
			
		||||
            catch
 | 
			
		||||
            {
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public override bool IsConnected() => _mqttClient?.IsConnected == true;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override string ToString()
 | 
			
		||||
    {
 | 
			
		||||
        return $" {nameof(MqttClient)} IP:{driverPropertys.IP} Port:{driverPropertys.Port}";
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            _globalDeviceData?.AllVariables?.ForEach(a => a.VariableValueChange -= VariableValueChange);
 | 
			
		||||
 | 
			
		||||
            _globalDeviceData?.CollectDevices?.ForEach(a =>
 | 
			
		||||
            {
 | 
			
		||||
                a.DeviceStatusChange -= DeviceStatusChange;
 | 
			
		||||
            });
 | 
			
		||||
            _mqttClient?.SafeDispose();
 | 
			
		||||
            _mqttClient = null;
 | 
			
		||||
            _uploadVariables = null;
 | 
			
		||||
            _collectDeviceRunTimes.Clear();
 | 
			
		||||
            _collectVariableRunTimes.Clear();
 | 
			
		||||
            _collectDeviceRunTimes = null;
 | 
			
		||||
            _collectVariableRunTimes = null;
 | 
			
		||||
            base.Dispose(disposing);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogError(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(UploadDeviceRunTime device)
 | 
			
		||||
    {
 | 
			
		||||
        var mqttFactory = new MqttFactory(new PrivateLogger(LogMessage));
 | 
			
		||||
        _mqttClientOptions = mqttFactory.CreateClientOptionsBuilder()
 | 
			
		||||
           .WithClientId(driverPropertys.ConnectId)
 | 
			
		||||
           .WithCredentials(driverPropertys.UserName, driverPropertys.Password)//账密
 | 
			
		||||
           .WithTcpServer(driverPropertys.IP, driverPropertys.Port)//服务器
 | 
			
		||||
           .WithCleanSession(true)
 | 
			
		||||
           .WithKeepAlivePeriod(TimeSpan.FromSeconds(120.0))
 | 
			
		||||
           .WithoutThrowOnNonSuccessfulConnectResponse()
 | 
			
		||||
           .Build();
 | 
			
		||||
        _mqttSubscribeOptions = mqttFactory.CreateSubscribeOptionsBuilder()
 | 
			
		||||
            .WithTopicFilter(
 | 
			
		||||
                f =>
 | 
			
		||||
                {
 | 
			
		||||
                    f.WithTopic(driverPropertys.RpcWriteTopic);
 | 
			
		||||
                })
 | 
			
		||||
                     .WithTopicFilter(
 | 
			
		||||
                f =>
 | 
			
		||||
                {
 | 
			
		||||
                    f.WithTopic(driverPropertys.QuestRpcTopic);
 | 
			
		||||
                })
 | 
			
		||||
            .Build();
 | 
			
		||||
        _mqttClient = mqttFactory.CreateMqttClient();
 | 
			
		||||
        _mqttClient.ConnectedAsync += MqttClient_ConnectedAsync;
 | 
			
		||||
        _mqttClient.ApplicationMessageReceivedAsync += MqttClient_ApplicationMessageReceivedAsync;
 | 
			
		||||
        _globalDeviceData = App.GetService<GlobalDeviceData>();
 | 
			
		||||
        _rpcCore = App.GetService<RpcSingletonService>();
 | 
			
		||||
        collectDeviceHostService = BackgroundServiceUtil.GetBackgroundService<CollectDeviceWorker>();
 | 
			
		||||
 | 
			
		||||
        var tags = _globalDeviceData.AllVariables.Where(a => a.VariablePropertys.ContainsKey(device.Id))
 | 
			
		||||
           .Where(b => GetPropertyValue(b, nameof(variablePropertys.Enable)).GetBoolValue())
 | 
			
		||||
           .ToList();
 | 
			
		||||
 | 
			
		||||
        _uploadVariables = tags;
 | 
			
		||||
 | 
			
		||||
        _collectDevice = _globalDeviceData.CollectDevices.Where(a => _uploadVariables.Select(b => b.DeviceId).Contains(a.Id)).ToList();
 | 
			
		||||
        if (!driverPropertys.IsInterval)
 | 
			
		||||
        {
 | 
			
		||||
            _collectDevice.ForEach(a =>
 | 
			
		||||
            {
 | 
			
		||||
                a.DeviceStatusChange += DeviceStatusChange;
 | 
			
		||||
            });
 | 
			
		||||
            _uploadVariables.ForEach(a =>
 | 
			
		||||
            {
 | 
			
		||||
                a.VariableValueChange += VariableValueChange;
 | 
			
		||||
            });
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (driverPropertys.UploadInterval <= 1000) driverPropertys.UploadInterval = 1000;
 | 
			
		||||
        exVariableTimerTick = new(driverPropertys.UploadInterval);
 | 
			
		||||
        exDeviceTimerTick = new(driverPropertys.UploadInterval);
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task AllPublishAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        //保留消息
 | 
			
		||||
        //分解List,避免超出mqtt字节大小限制
 | 
			
		||||
        var varData = _globalDeviceData.AllVariables.Adapt<List<VariableData>>().ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
        var devData = _globalDeviceData.CollectDevices.Adapt<List<DeviceData>>().ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
        var isConnect = await TryMqttClientAsync(cancellationToken);
 | 
			
		||||
        foreach (var item in devData)
 | 
			
		||||
        {
 | 
			
		||||
            var devMessage = new MqttApplicationMessageBuilder()
 | 
			
		||||
.WithTopic($"{driverPropertys.DeviceTopic}")
 | 
			
		||||
.WithPayload(item.GetSciptListValue(driverPropertys.BigTextScriptDeviceModel)).Build();
 | 
			
		||||
            if (isConnect.IsSuccess)
 | 
			
		||||
                await _mqttClient.PublishAsync(devMessage, cancellationToken);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        foreach (var item in varData)
 | 
			
		||||
        {
 | 
			
		||||
            var varMessage = new MqttApplicationMessageBuilder()
 | 
			
		||||
            .WithTopic($"{driverPropertys.VariableTopic}")
 | 
			
		||||
            .WithPayload(item.GetSciptListValue(driverPropertys.BigTextScriptVariableModel)).Build();
 | 
			
		||||
            if (isConnect.IsSuccess)
 | 
			
		||||
                await _mqttClient.PublishAsync(varMessage, cancellationToken);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void DeviceStatusChange(CollectDeviceRunTime collectDeviceRunTime)
 | 
			
		||||
    {
 | 
			
		||||
        if (driverPropertys?.IsInterval != true)
 | 
			
		||||
            _collectDeviceRunTimes.Enqueue(collectDeviceRunTime.Adapt<DeviceData>());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task MqttClient_ApplicationMessageReceivedAsync(MqttApplicationMessageReceivedEventArgs args)
 | 
			
		||||
    {
 | 
			
		||||
        if (args.ApplicationMessage.Topic == driverPropertys.QuestRpcTopic && args.ApplicationMessage.PayloadSegment.Count > 0)
 | 
			
		||||
        {
 | 
			
		||||
            await AllPublishAsync(CancellationToken.None);
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (!driverPropertys.DeviceRpcEnable || string.IsNullOrEmpty(args.ClientId))
 | 
			
		||||
            return;
 | 
			
		||||
        if (args.ApplicationMessage.Topic != driverPropertys.RpcWriteTopic)
 | 
			
		||||
            return;
 | 
			
		||||
        var rpcDatas = Encoding.UTF8.GetString(args.ApplicationMessage.PayloadSegment).FromJsonString<MqttRpcNameVaueWithId>();
 | 
			
		||||
        if (rpcDatas == null)
 | 
			
		||||
            return;
 | 
			
		||||
 | 
			
		||||
        MqttRpcResult mqttRpcResult = new() { RpcId = rpcDatas.RpcId, Success = true };
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            foreach (var rpcData in rpcDatas.WriteInfos)
 | 
			
		||||
            {
 | 
			
		||||
 | 
			
		||||
                var tag = _uploadVariables.FirstOrDefault(a => a.Name == rpcData.Key);
 | 
			
		||||
                if (tag != null)
 | 
			
		||||
                {
 | 
			
		||||
                    var rpcEnable = GetPropertyValue(tag, nameof(variablePropertys.VariableRpcEnable)).ToBoolean();
 | 
			
		||||
                    if (rpcEnable == true)
 | 
			
		||||
                    {
 | 
			
		||||
 | 
			
		||||
                    }
 | 
			
		||||
                    else
 | 
			
		||||
                    {
 | 
			
		||||
                        mqttRpcResult.Success = false;
 | 
			
		||||
                        mqttRpcResult.Message.Add(rpcData.Key, new("权限不足,变量不支持写入"));
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    mqttRpcResult.Success = false;
 | 
			
		||||
                    mqttRpcResult.Message.Add(rpcData.Key, new("不存在该变量"));
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            var result = await _rpcCore.InvokeDeviceMethodAsync(ToString() + "-" + args.ClientId,
 | 
			
		||||
                rpcDatas.WriteInfos.Where(
 | 
			
		||||
                a => !mqttRpcResult.Message.Any(b => b.Key == a.Key)).ToDictionary(a => a.Key, a => a.Value));
 | 
			
		||||
 | 
			
		||||
            mqttRpcResult.Message.AddRange(result);
 | 
			
		||||
            mqttRpcResult.Success = !mqttRpcResult.Message.Any(a => !a.Value.IsSuccess);
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var variableMessage = new MqttApplicationMessageBuilder()
 | 
			
		||||
.WithTopic($"{driverPropertys.RpcSubTopic}")
 | 
			
		||||
.WithPayload(mqttRpcResult.ToJsonString()).Build();
 | 
			
		||||
            var isConnect = await TryMqttClientAsync(CancellationToken.None);
 | 
			
		||||
            if (isConnect.IsSuccess)
 | 
			
		||||
                await _mqttClient.PublishAsync(variableMessage);
 | 
			
		||||
        }
 | 
			
		||||
        catch
 | 
			
		||||
        {
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task MqttClient_ConnectedAsync(MqttClientConnectedEventArgs arg)
 | 
			
		||||
    {
 | 
			
		||||
        var subResult = await _mqttClient.SubscribeAsync(_mqttSubscribeOptions);
 | 
			
		||||
        if (subResult.Items.Any(a => a.ResultCode > (MqttClientSubscribeResultCode)10))
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.Warning($"订阅失败-{subResult.Items
 | 
			
		||||
                .Where(a => a.ResultCode > (MqttClientSubscribeResultCode)10)
 | 
			
		||||
                .Select(a =>
 | 
			
		||||
                new
 | 
			
		||||
                {
 | 
			
		||||
                    Topic = a.TopicFilter.Topic,
 | 
			
		||||
                    ResultCode = a.ResultCode.ToString()
 | 
			
		||||
                }
 | 
			
		||||
                )
 | 
			
		||||
                .ToJsonString()}");
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 上传mqtt内容,并进行离线缓存
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="topic"></param>
 | 
			
		||||
    /// <param name="payLoad"></param>
 | 
			
		||||
    /// <param name="cancellationToken"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    private async Task MqttUp(string topic, string payLoad, CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        var variableMessage = new MqttApplicationMessageBuilder()
 | 
			
		||||
.WithTopic(topic)
 | 
			
		||||
.WithPayload(payLoad).Build();
 | 
			
		||||
        var isConnect = await TryMqttClientAsync(cancellationToken);
 | 
			
		||||
        if (isConnect.IsSuccess)
 | 
			
		||||
        {
 | 
			
		||||
            //连接成功时补发缓存数据
 | 
			
		||||
            var cacheData = await CacheDb.GetCacheData();
 | 
			
		||||
            foreach (var item in cacheData)
 | 
			
		||||
            {
 | 
			
		||||
                var cacheMessage = new MqttApplicationMessageBuilder()
 | 
			
		||||
.WithTopic(item.Topic)
 | 
			
		||||
.WithPayload(item.CacheStr).Build();
 | 
			
		||||
                var cacheResult = await _mqttClient.PublishAsync(cacheMessage, cancellationToken);
 | 
			
		||||
                if (cacheResult.IsSuccess)
 | 
			
		||||
                {
 | 
			
		||||
                    await CacheDb.DeleteCacheData(item.Id);
 | 
			
		||||
                    LogMessage.Trace($"{FoundationConst.LogMessageHeader}主题:{item.Topic}{Environment.NewLine}负载:{item.CacheStr}");
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            var result = await _mqttClient.PublishAsync(variableMessage, cancellationToken);
 | 
			
		||||
            if (!result.IsSuccess)
 | 
			
		||||
            {
 | 
			
		||||
                await CacheDb.AddCacheData(topic, payLoad, driverPropertys.CacheMaxCount);
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                LogMessage.Trace($"{FoundationConst.LogMessageHeader}主题:{topic}{Environment.NewLine}负载:{payLoad}");
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
            await CacheDb.AddCacheData(topic, payLoad, driverPropertys.CacheMaxCount);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    private async Task<OperResult> TryMqttClientAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        if (_mqttClient?.IsConnected == true)
 | 
			
		||||
            return OperResult.CreateSuccessResult();
 | 
			
		||||
        return await Cilent();
 | 
			
		||||
 | 
			
		||||
        async Task<OperResult> Cilent()
 | 
			
		||||
        {
 | 
			
		||||
            if (_mqttClient?.IsConnected == true)
 | 
			
		||||
                return OperResult.CreateSuccessResult();
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                await easyLock.WaitAsync();
 | 
			
		||||
                if (_mqttClient?.IsConnected == true)
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
                using var timeoutToken = new CancellationTokenSource(TimeSpan.FromMilliseconds(driverPropertys.ConnectTimeOut));
 | 
			
		||||
                using CancellationTokenSource StoppingToken = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken, timeoutToken.Token);
 | 
			
		||||
                if (_mqttClient?.IsConnected == true)
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
                if (_mqttClient == null)
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult("未初始化");
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
                var result = await _mqttClient?.ConnectAsync(_mqttClientOptions, StoppingToken.Token);
 | 
			
		||||
                if (result.ResultCode == MqttClientConnectResultCode.Success)
 | 
			
		||||
                {
 | 
			
		||||
                    return OperResult.CreateSuccessResult();
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    return new OperResult(result.ReasonString);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            catch (Exception ex)
 | 
			
		||||
            {
 | 
			
		||||
                return new OperResult(ex);
 | 
			
		||||
            }
 | 
			
		||||
            finally
 | 
			
		||||
            {
 | 
			
		||||
                easyLock.Release();
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void VariableValueChange(DeviceVariableRunTime collectVariableRunTime)
 | 
			
		||||
    {
 | 
			
		||||
        if (driverPropertys?.IsInterval != true)
 | 
			
		||||
            _collectVariableRunTimes.Enqueue(collectVariableRunTime.Adapt<VariableData>());
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,85 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@page "/MqttClient"
 | 
			
		||||
 | 
			
		||||
@namespace ThingsGateway.Foundation.Demo
 | 
			
		||||
@using BlazorComponent;
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web;
 | 
			
		||||
@using Masa.Blazor.Presets;
 | 
			
		||||
@using Microsoft.JSInterop;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Extension;
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
<MqttClientPage @ref=mqttClientPage></MqttClientPage>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
<DriverDebugUIPage @ref=driverDebugUIPage Sections="_sections" ShowDefaultOtherContent=false>
 | 
			
		||||
    <ReadWriteContent>
 | 
			
		||||
        <div class="my-1 py-1">
 | 
			
		||||
 | 
			
		||||
            <MTextarea Class="mx-1 my-1" Label="订阅主题" Dense Outlined HideDetails="@("auto")" @bind-Value=@driverDebugUIPage.Address />
 | 
			
		||||
 | 
			
		||||
             <MButton Class="mx-1 my-1" Color="primary" OnClick="SubscribeAsync">
 | 
			
		||||
                添加
 | 
			
		||||
            </MButton>
 | 
			
		||||
             <MButton Class="mx-1 my-1" Color="primary" OnClick="UnsubscribeAsync">
 | 
			
		||||
                移除
 | 
			
		||||
            </MButton>
 | 
			
		||||
 | 
			
		||||
             <MTextarea Class="mx-1 my-1" Label="发布主题" Dense Outlined HideDetails="@("auto")" @bind-Value=@PublishTopic />
 | 
			
		||||
             <MTextarea Class="mx-1 my-1" Label="发布内容" Dense Outlined HideDetails="@("auto")" @bind-Value=@PublishValue />
 | 
			
		||||
 | 
			
		||||
             <MButton Class="mx-1 my-1" Color="primary" OnClick="PublishAsync">
 | 
			
		||||
                 发布
 | 
			
		||||
             </MButton>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
             <MSubheader Class="mt-4 font-weight-black"> Rpc  </MSubheader>
 | 
			
		||||
 | 
			
		||||
             <MTextarea Class="mx-1 mt-3 my-1" Label="Rpc主题" Dense Outlined HideDetails="@("auto")" @bind-Value=@MqttRpcTopicPair.RequestTopic />
 | 
			
		||||
             <MTextarea Class="mx-1 mt-3 my-1" Label="Rpc返回主题" Dense Outlined HideDetails="@("auto")" @bind-Value=@MqttRpcTopicPair.ResponseTopic />
 | 
			
		||||
             <MTextarea Class="mx-1 mt-3 my-1" Label="Rpc内容" Dense Outlined HideDetails="@("auto")" @bind-Value=@driverDebugUIPage.WriteValue />
 | 
			
		||||
             <MButton Class="mx-1 my-1" Color="primary" OnClick="RpcExecuteAsync">
 | 
			
		||||
                执行
 | 
			
		||||
            </MButton>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        <MCol Class="my-1 py-1">
 | 
			
		||||
            <MRow NoGutters>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
            </MRow>
 | 
			
		||||
 | 
			
		||||
        </MCol>
 | 
			
		||||
 | 
			
		||||
    </ReadWriteContent>
 | 
			
		||||
</DriverDebugUIPage>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@code {
 | 
			
		||||
    private readonly List<(string Code, string Language)> _sections = new();
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,152 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using Masa.Blazor;
 | 
			
		||||
 | 
			
		||||
using Microsoft.AspNetCore.Components;
 | 
			
		||||
 | 
			
		||||
using MQTTnet;
 | 
			
		||||
using MQTTnet.Extensions.Rpc;
 | 
			
		||||
 | 
			
		||||
using System.Collections.Generic;
 | 
			
		||||
using System.Text;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Plugin.Mqtt;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Demo;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// MqttClientDebugPage
 | 
			
		||||
/// </summary>
 | 
			
		||||
public partial class MqttClientDebugPage : IDisposable
 | 
			
		||||
{
 | 
			
		||||
    private DriverDebugUIPage driverDebugUIPage;
 | 
			
		||||
    private MqttClientPage mqttClientPage;
 | 
			
		||||
 | 
			
		||||
    [Inject]
 | 
			
		||||
    IPopupService PopupService { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public void Dispose()
 | 
			
		||||
    {
 | 
			
		||||
        mqttClientPage.SafeDispose();
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void OnAfterRender(bool firstRender)
 | 
			
		||||
    {
 | 
			
		||||
        if (firstRender)
 | 
			
		||||
        {
 | 
			
		||||
            if (mqttClientPage != null)
 | 
			
		||||
            {
 | 
			
		||||
                mqttClientPage.LogAction = driverDebugUIPage.LogOut;
 | 
			
		||||
            }
 | 
			
		||||
            //初始化
 | 
			
		||||
            driverDebugUIPage.Address = "ThingsGateway/Variable";
 | 
			
		||||
 | 
			
		||||
            driverDebugUIPage.WriteValue = new MqttRpcNameVaueWithId()
 | 
			
		||||
            {
 | 
			
		||||
                RpcId = Guid.NewGuid().ToString(),
 | 
			
		||||
                WriteInfos = new Dictionary<string, string>()
 | 
			
		||||
{
 | 
			
		||||
    { "tag1", "123" }
 | 
			
		||||
}
 | 
			
		||||
            }.ToJsonString();
 | 
			
		||||
            ;
 | 
			
		||||
            mqttClientPage.IP = "127.0.0.1";
 | 
			
		||||
            mqttClientPage.Port = 1883;
 | 
			
		||||
            mqttClientPage.UserName = "admin";
 | 
			
		||||
            mqttClientPage.Password = "111111";
 | 
			
		||||
            mqttClientPage.StateHasChangedAsync();
 | 
			
		||||
 | 
			
		||||
            //载入配置
 | 
			
		||||
            StateHasChanged();
 | 
			
		||||
            driverDebugUIPage.Sections.Clear();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        base.OnAfterRender(firstRender);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task SubscribeAsync()
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var mqttSubscribeOptions = mqttClientPage.MqttFactory.CreateSubscribeOptionsBuilder()
 | 
			
		||||
.WithTopicFilter(
 | 
			
		||||
f =>
 | 
			
		||||
{
 | 
			
		||||
    f.WithTopic(driverDebugUIPage.Address);
 | 
			
		||||
})
 | 
			
		||||
.Build();
 | 
			
		||||
 | 
			
		||||
            await mqttClientPage.MqttClient.SubscribeAsync(mqttSubscribeOptions, CancellationToken.None);
 | 
			
		||||
            driverDebugUIPage.Messages.Add((Microsoft.Extensions.Logging.LogLevel.Information, DateTimeExtensions.CurrentDateTime.ToDefaultDateTimeFormat(driverDebugUIPage.InitTimezone.TimezoneOffset) + " - " + $"订阅{driverDebugUIPage.Address}成功"));
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            driverDebugUIPage.Messages.Add((Microsoft.Extensions.Logging.LogLevel.Error, DateTimeExtensions.CurrentDateTime.ToDefaultDateTimeFormat(driverDebugUIPage.InitTimezone.TimezoneOffset) + " - " + ex.Message));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    private async Task UnsubscribeAsync()
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var mqttSubscribeOptions = mqttClientPage.MqttFactory.CreateUnsubscribeOptionsBuilder()
 | 
			
		||||
.WithTopicFilter(driverDebugUIPage.Address)
 | 
			
		||||
.Build();
 | 
			
		||||
 | 
			
		||||
            await mqttClientPage.MqttClient.UnsubscribeAsync(mqttSubscribeOptions, CancellationToken.None);
 | 
			
		||||
            driverDebugUIPage.Messages.Add((Microsoft.Extensions.Logging.LogLevel.Information, DateTimeExtensions.CurrentDateTime.ToDefaultDateTimeFormat(driverDebugUIPage.InitTimezone.TimezoneOffset) + " - " + $"取消订阅{driverDebugUIPage.Address}成功"));
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
            driverDebugUIPage.Messages.Add((Microsoft.Extensions.Logging.LogLevel.Error, DateTimeExtensions.CurrentDateTime.ToDefaultDateTimeFormat(driverDebugUIPage.InitTimezone.TimezoneOffset) + " - " + ex.Message));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    string PublishTopic;
 | 
			
		||||
    string PublishValue;
 | 
			
		||||
    private async Task PublishAsync()
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var devMessage = new MqttApplicationMessageBuilder()
 | 
			
		||||
.WithTopic($"{PublishTopic}")
 | 
			
		||||
.WithPayload(PublishValue).Build();
 | 
			
		||||
            await mqttClientPage.MqttClient.PublishAsync(devMessage, CancellationToken.None);
 | 
			
		||||
            driverDebugUIPage.Messages.Add((Microsoft.Extensions.Logging.LogLevel.Information, DateTimeExtensions.CurrentDateTime.ToDefaultDateTimeFormat(driverDebugUIPage.InitTimezone.TimezoneOffset) + " - " + $"发布{PublishTopic}成功"));
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            driverDebugUIPage.Messages.Add((Microsoft.Extensions.Logging.LogLevel.Error, DateTimeExtensions.CurrentDateTime.ToDefaultDateTimeFormat(driverDebugUIPage.InitTimezone.TimezoneOffset) + " - " + ex.Message));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    MqttRpcTopicPair MqttRpcTopicPair = new() { RequestTopic = "ThingsGateway/RpcWrite", ResponseTopic = "ThingsGateway/RpcSub" };
 | 
			
		||||
    private async Task RpcExecuteAsync()
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            using MqttRpcClient mqttRpcClient = new(mqttClientPage.MqttClient);
 | 
			
		||||
            var data = await mqttRpcClient.ExecuteAsync(MqttRpcTopicPair, driverDebugUIPage.WriteValue, MQTTnet.Protocol.MqttQualityOfServiceLevel.AtMostOnce, TimeSpan.FromSeconds(10));
 | 
			
		||||
            var str = Encoding.UTF8.GetString(data);
 | 
			
		||||
            driverDebugUIPage.Messages.Add((Microsoft.Extensions.Logging.LogLevel.Information, DateTimeExtensions.CurrentDateTime.ToDefaultDateTimeFormat(driverDebugUIPage.InitTimezone.TimezoneOffset) + " - " + str));
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            driverDebugUIPage.Messages.Add((Microsoft.Extensions.Logging.LogLevel.Error, DateTimeExtensions.CurrentDateTime.ToDefaultDateTimeFormat(driverDebugUIPage.InitTimezone.TimezoneOffset) + " - " + ex.Message));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,119 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using MQTTnet;
 | 
			
		||||
using MQTTnet.Client;
 | 
			
		||||
 | 
			
		||||
using System.Text;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Plugin.Mqtt;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Demo;
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// MqttClientPage
 | 
			
		||||
/// </summary>
 | 
			
		||||
public partial class MqttClientPage
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// 日志输出
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public Action<LogLevel, object, string, Exception> LogAction;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// OPC
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public IMqttClient MqttClient;
 | 
			
		||||
    public MqttClientOptions MqttClientOptions;
 | 
			
		||||
 | 
			
		||||
    public MqttFactory MqttFactory;
 | 
			
		||||
 | 
			
		||||
    public string ConnectId;
 | 
			
		||||
 | 
			
		||||
    public string IP;
 | 
			
		||||
 | 
			
		||||
    public string Password;
 | 
			
		||||
 | 
			
		||||
    public int Port;
 | 
			
		||||
 | 
			
		||||
    public string UserName;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public void Dispose()
 | 
			
		||||
    {
 | 
			
		||||
        MqttClient.SafeDispose();
 | 
			
		||||
    }
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void OnInitialized()
 | 
			
		||||
    {
 | 
			
		||||
        MqttFactory = new MqttFactory(new PrivateLogger(new EasyLogger(LogAction) { LogLevel = LogLevel.Trace }));
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        MqttClient = MqttFactory.CreateMqttClient();
 | 
			
		||||
        MqttClient.ApplicationMessageReceivedAsync += MqttClient_ApplicationMessageReceivedAsync;
 | 
			
		||||
        base.OnInitialized();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task Connect()
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await MqttClient.DisconnectAsync();
 | 
			
		||||
            await GetMqttClient();
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogAction?.Invoke(LogLevel.Error, null, null, ex);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task DisConnect()
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            await MqttClient.DisconnectAsync();
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogAction?.Invoke(LogLevel.Error, null, null, ex);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task<IMqttClient> GetMqttClient()
 | 
			
		||||
    {
 | 
			
		||||
        //载入配置
 | 
			
		||||
        MqttClientOptions = MqttFactory.CreateClientOptionsBuilder()
 | 
			
		||||
   .WithClientId(ConnectId)
 | 
			
		||||
   .WithCredentials(UserName, Password)//账密
 | 
			
		||||
   .WithTcpServer(IP, Port)//服务器
 | 
			
		||||
   .WithCleanSession(true)
 | 
			
		||||
   .WithKeepAlivePeriod(TimeSpan.FromSeconds(120.0))
 | 
			
		||||
   .WithoutThrowOnNonSuccessfulConnectResponse()
 | 
			
		||||
   .Build();
 | 
			
		||||
        await MqttClient.ConnectAsync(MqttClientOptions);
 | 
			
		||||
        return MqttClient;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void LogOut(byte logLevel, object source, string message, Exception exception) => LogAction?.Invoke((LogLevel)logLevel, source, message, exception);
 | 
			
		||||
 | 
			
		||||
    private Task MqttClient_ApplicationMessageReceivedAsync(MqttApplicationMessageReceivedEventArgs args)
 | 
			
		||||
    {
 | 
			
		||||
        LogAction?.Invoke(LogLevel.Info, this, $"[{args.ApplicationMessage.Topic}]:{Encoding.UTF8.GetString(args.ApplicationMessage.PayloadSegment)}", null);
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public Task StateHasChangedAsync()
 | 
			
		||||
    {
 | 
			
		||||
        return InvokeAsync(StateHasChanged);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,49 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using MQTTnet.Diagnostics;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Mqtt
 | 
			
		||||
{
 | 
			
		||||
    internal class PrivateLogger : IMqttNetLogger
 | 
			
		||||
    {
 | 
			
		||||
        readonly ILog LogMessage;
 | 
			
		||||
        public PrivateLogger(ILog logger)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage = logger;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        public bool IsEnabled => true;
 | 
			
		||||
        public void Publish(MqttNetLogLevel logLevel, string source, string message, object[] parameters, Exception exception)
 | 
			
		||||
        {
 | 
			
		||||
            switch (logLevel)
 | 
			
		||||
            {
 | 
			
		||||
                case MqttNetLogLevel.Verbose:
 | 
			
		||||
                    LogMessage?.Log(LogLevel.Trace, source, message != null ? (parameters != null ? message != null ? (parameters != null ? string.Format(message, parameters) : message) : string.Empty : message) : string.Empty, exception);
 | 
			
		||||
                    break;
 | 
			
		||||
 | 
			
		||||
                case MqttNetLogLevel.Info:
 | 
			
		||||
                    LogMessage?.Log(LogLevel.Info, source, message != null ? (parameters != null ? string.Format(message, parameters) : message) : string.Empty, exception);
 | 
			
		||||
                    break;
 | 
			
		||||
 | 
			
		||||
                case MqttNetLogLevel.Warning:
 | 
			
		||||
                    LogMessage?.Log(LogLevel.Warning, source, message != null ? (parameters != null ? string.Format(message, parameters) : message) : string.Empty, exception);
 | 
			
		||||
                    break;
 | 
			
		||||
 | 
			
		||||
                case MqttNetLogLevel.Error:
 | 
			
		||||
                    LogMessage?.Log(LogLevel.Warning, source, message != null ? (parameters != null ? string.Format(message, parameters) : message) : string.Empty, exception);
 | 
			
		||||
                    break;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,34 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  <20>˴<EFBFBD><CBB4><EFBFBD><EFBFBD><EFBFBD>Ȩ<EFBFBD><C8A8><EFBFBD><EFBFBD>Ϊȫ<CEAA>ļ<EFBFBD><C4BC><EFBFBD><EFBFBD>ǣ<EFBFBD><C7A3><EFBFBD><EFBFBD><EFBFBD>ԭ<EFBFBD><D4AD><EFBFBD><EFBFBD><EFBFBD>ر<EFBFBD><D8B1><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>·<EFBFBD><C2B7>ֶ<EFBFBD><D6B6><EFBFBD><EFBFBD><EFBFBD>
 | 
			
		||||
//  <20>˴<EFBFBD><CBB4><EFBFBD><EFBFBD><EFBFBD>Ȩ<EFBFBD><C8A8><EFBFBD><EFBFBD><EFBFBD>ر<EFBFBD><D8B1><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ĵ<EFBFBD><C4B4>룩<EFBFBD><EBA3A9><EFBFBD><EFBFBD><EFBFBD>߱<EFBFBD><DFB1><EFBFBD>Diego<67><6F><EFBFBD><EFBFBD>
 | 
			
		||||
//  Դ<><D4B4><EFBFBD><EFBFBD>ʹ<EFBFBD><CAB9>Э<EFBFBD><D0AD><EFBFBD><EFBFBD>ѭ<EFBFBD><D1AD><EFBFBD>ֿ<EFBFBD><D6BF>Ŀ<EFBFBD>ԴЭ<D4B4>鼰<EFBFBD><E9BCB0><EFBFBD><EFBFBD>Э<EFBFBD><D0AD>
 | 
			
		||||
//  GiteeԴ<65><D4B4><EFBFBD><EFBFBD><EFBFBD>ֿ⣺https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  GithubԴ<62><D4B4><EFBFBD><EFBFBD><EFBFBD>ֿ⣺https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  ʹ<><CAB9><EFBFBD>ĵ<EFBFBD><C4B5><EFBFBD>https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQȺ<51><C8BA>605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
// Licensed to the .NET Foundation under one or more agreements.
 | 
			
		||||
// The .NET Foundation licenses this file to you under the MIT license.
 | 
			
		||||
// See the LICENSE file in the project root for more information.
 | 
			
		||||
 | 
			
		||||
using MQTTnet.Protocol;
 | 
			
		||||
 | 
			
		||||
using System.Text;
 | 
			
		||||
 | 
			
		||||
namespace MQTTnet.Extensions.Rpc
 | 
			
		||||
{
 | 
			
		||||
    public static class MqttRpcClientExtensions
 | 
			
		||||
    {
 | 
			
		||||
        public static Task<byte[]> ExecuteAsync(this MqttRpcClient client, MqttRpcTopicPair mqttRpcTopicPair, string payload, MqttQualityOfServiceLevel qualityOfServiceLevel, TimeSpan timeout)
 | 
			
		||||
        {
 | 
			
		||||
            if (client == null) throw new ArgumentNullException(nameof(client));
 | 
			
		||||
 | 
			
		||||
            var buffer = Encoding.UTF8.GetBytes(payload ?? string.Empty);
 | 
			
		||||
 | 
			
		||||
            return client.ExecuteAsync(mqttRpcTopicPair, buffer, qualityOfServiceLevel, timeout);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,25 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  <20>˴<EFBFBD><CBB4><EFBFBD><EFBFBD><EFBFBD>Ȩ<EFBFBD><C8A8><EFBFBD><EFBFBD>Ϊȫ<CEAA>ļ<EFBFBD><C4BC><EFBFBD><EFBFBD>ǣ<EFBFBD><C7A3><EFBFBD><EFBFBD><EFBFBD>ԭ<EFBFBD><D4AD><EFBFBD><EFBFBD><EFBFBD>ر<EFBFBD><D8B1><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>·<EFBFBD><C2B7>ֶ<EFBFBD><D6B6><EFBFBD><EFBFBD><EFBFBD>
 | 
			
		||||
//  <20>˴<EFBFBD><CBB4><EFBFBD><EFBFBD><EFBFBD>Ȩ<EFBFBD><C8A8><EFBFBD><EFBFBD><EFBFBD>ر<EFBFBD><D8B1><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ĵ<EFBFBD><C4B4>룩<EFBFBD><EBA3A9><EFBFBD><EFBFBD><EFBFBD>߱<EFBFBD><DFB1><EFBFBD>Diego<67><6F><EFBFBD><EFBFBD>
 | 
			
		||||
//  Դ<><D4B4><EFBFBD><EFBFBD>ʹ<EFBFBD><CAB9>Э<EFBFBD><D0AD><EFBFBD><EFBFBD>ѭ<EFBFBD><D1AD><EFBFBD>ֿ<EFBFBD><D6BF>Ŀ<EFBFBD>ԴЭ<D4B4>鼰<EFBFBD><E9BCB0><EFBFBD><EFBFBD>Э<EFBFBD><D0AD>
 | 
			
		||||
//  GiteeԴ<65><D4B4><EFBFBD><EFBFBD><EFBFBD>ֿ⣺https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  GithubԴ<62><D4B4><EFBFBD><EFBFBD><EFBFBD>ֿ⣺https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  ʹ<><CAB9><EFBFBD>ĵ<EFBFBD><C4B5><EFBFBD>https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQȺ<51><C8BA>605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
// Licensed to the .NET Foundation under one or more agreements.
 | 
			
		||||
// The .NET Foundation licenses this file to you under the MIT license.
 | 
			
		||||
// See the LICENSE file in the project root for more information.
 | 
			
		||||
 | 
			
		||||
namespace MQTTnet.Extensions.Rpc
 | 
			
		||||
{
 | 
			
		||||
    public sealed class MqttRpcTopicPair
 | 
			
		||||
    {
 | 
			
		||||
        public string RequestTopic { get; set; }
 | 
			
		||||
 | 
			
		||||
        public string ResponseTopic { get; set; }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,37 +0,0 @@
 | 
			
		||||
<Project Sdk="Microsoft.NET.Sdk.Razor">
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	<Target Name="PostBuild" AfterTargets="PostBuildEvent">
 | 
			
		||||
 | 
			
		||||
		<Exec Command=" set dir="$(SolutionDir)Web\ThingsGateway.Web.Entry\bin\$(Configuration)\$(TargetFramework)\Plugins\$(AssemblyName)"
 if not exist %25dir%25  md %25dir%25  
copy "$(TargetDir)*Mqtt*.dll"  %25dir%25


" />
 | 
			
		||||
	
 | 
			
		||||
	</Target>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
		<ProjectReference Include="..\..\Web\ThingsGateway.Components\ThingsGateway.Components.csproj">
 | 
			
		||||
			<Private>false</Private>
 | 
			
		||||
			<CopyLocalSatelliteAssemblies>false</CopyLocalSatelliteAssemblies>
 | 
			
		||||
			<ExcludeAssets>runtime</ExcludeAssets>
 | 
			
		||||
		</ProjectReference>
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\DriverDebugUIBase.cs" Link="DebugPage\DriverDebugUIBase.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\DriverDebugUIPage.razor.cs" Link="DebugPage\DriverDebugUIPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\SerialSessionPage.razor.cs" Link="DebugPage\SerialSessionPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpClientPage.razor.cs" Link="DebugPage\TcpClientPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpServerPage.razor.cs" Link="DebugPage\TcpServerPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\UdpSessionPage.razor.cs" Link="DebugPage\UdpSessionPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\DriverDebugUIPage.razor" Link="DebugPage\DriverDebugUIPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\SerialSessionPage.razor" Link="DebugPage\SerialSessionPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpClientPage.razor" Link="DebugPage\TcpClientPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpServerPage.razor" Link="DebugPage\TcpServerPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\UdpSessionPage.razor" Link="DebugPage\UdpSessionPage.razor" />
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
	
 | 
			
		||||
 | 
			
		||||
</Project>
 | 
			
		||||
@@ -1,46 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@namespace ThingsGateway.Foundation.Demo
 | 
			
		||||
@using BlazorComponent;
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web;
 | 
			
		||||
@using System.IO.Ports;
 | 
			
		||||
@using System.Collections.Concurrent;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Adapter.OPCDA;
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
@implements IDisposable
 | 
			
		||||
<MCard Elevation="1" Rounded="false" Class="pa-2" Style="width:100%">
 | 
			
		||||
    <div class="mb-4">设备配置</div>
 | 
			
		||||
    <MRow Justify="JustifyTypes.Start" Align="AlignTypes.Center">
 | 
			
		||||
        <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.GroupSize) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.GroupSize />
 | 
			
		||||
        <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.UpdateRate) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.UpdateRate />
 | 
			
		||||
 | 
			
		||||
        <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.DeadBand) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.DeadBand />
 | 
			
		||||
        <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.CheckRate) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.CheckRate />
 | 
			
		||||
        <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.OPCIP) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.OPCIP />
 | 
			
		||||
        <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.OPCName) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.OPCName />
 | 
			
		||||
 | 
			
		||||
        <MCheckbox Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.ActiveSubscribe) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.ActiveSubscribe />
 | 
			
		||||
 | 
			
		||||
        <MButton Class="ma-1" OnClick=@Connect Color="primary">
 | 
			
		||||
            连接
 | 
			
		||||
        </MButton>
 | 
			
		||||
 | 
			
		||||
        <MButton Class="ma-1" OnClick=@DisConnect Color="red">
 | 
			
		||||
            断开
 | 
			
		||||
        </MButton>
 | 
			
		||||
    </MRow>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
</MCard>
 | 
			
		||||
@@ -1,43 +0,0 @@
 | 
			
		||||
<Project Sdk="Microsoft.NET.Sdk.Razor">
 | 
			
		||||
 | 
			
		||||
	<PropertyGroup>
 | 
			
		||||
	  <DefineConstants>Plugin</DefineConstants>
 | 
			
		||||
	</PropertyGroup>
 | 
			
		||||
 | 
			
		||||
	<Target Name="PostBuild" AfterTargets="PostBuildEvent">
 | 
			
		||||
		<Exec Command=" set dir="$(SolutionDir)Web\ThingsGateway.Web.Entry\bin\$(Configuration)\$(TargetFramework)\Plugins\$(AssemblyName)"
 if not exist %25dir%25  md %25dir%25  
copy "$(TargetDir)*OPCDA*.dll"  %25dir%25


" />
 | 
			
		||||
	</Target>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\DriverDebugUIBase.cs" Link="DebugPage\DriverDebugUIBase.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\DriverDebugUIPage.razor.cs" Link="DebugPage\DriverDebugUIPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\SerialSessionPage.razor.cs" Link="DebugPage\SerialSessionPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpClientPage.razor.cs" Link="DebugPage\TcpClientPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpServerPage.razor.cs" Link="DebugPage\TcpServerPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\UdpSessionPage.razor.cs" Link="DebugPage\UdpSessionPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\DriverDebugUIPage.razor" Link="DebugPage\DriverDebugUIPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\SerialSessionPage.razor" Link="DebugPage\SerialSessionPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpClientPage.razor" Link="DebugPage\TcpClientPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpServerPage.razor" Link="DebugPage\TcpServerPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\UdpSessionPage.razor" Link="DebugPage\UdpSessionPage.razor" />
 | 
			
		||||
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
	
 | 
			
		||||
 
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
		<ProjectReference Include="..\..\Foundation\ThingsGateway.Foundation.Adapter.OPCDA\ThingsGateway.Foundation.Adapter.OPCDA.csproj" />
 | 
			
		||||
		<ProjectReference Include="..\..\Web\ThingsGateway.Components\ThingsGateway.Components.csproj">
 | 
			
		||||
			<Private>false</Private>
 | 
			
		||||
			<CopyLocalSatelliteAssemblies>false</CopyLocalSatelliteAssemblies>
 | 
			
		||||
			<ExcludeAssets>runtime</ExcludeAssets>
 | 
			
		||||
		</ProjectReference>
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
</Project>
 | 
			
		||||
@@ -1,35 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@using System.Net.Http
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Forms
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Routing
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web.Virtualization
 | 
			
		||||
@using Microsoft.JSInterop
 | 
			
		||||
@using BlazorComponent
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
@using Masa.Blazor.Presets
 | 
			
		||||
@using System.Net.Http.Json
 | 
			
		||||
@using System.IO;
 | 
			
		||||
@using System.Text.Json;
 | 
			
		||||
@using System.Reflection;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Components;
 | 
			
		||||
@using ThingsGateway.Admin.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Extension.String;
 | 
			
		||||
@using ThingsGateway.Admin.Application;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Core;
 | 
			
		||||
@using ThingsGateway.Gateway.Application;
 | 
			
		||||
@using ThingsGateway.Gateway.Core;
 | 
			
		||||
@@ -1,58 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@namespace ThingsGateway.Foundation.Demo
 | 
			
		||||
 | 
			
		||||
@using BlazorComponent;
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web;
 | 
			
		||||
@using System.IO.Ports;
 | 
			
		||||
@using System.Collections.Concurrent;
 | 
			
		||||
@using Opc.Ua.Client;
 | 
			
		||||
@using Opc.Ua;
 | 
			
		||||
@using System.Linq;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Adapter.OPCUA;
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
@implements IDisposable
 | 
			
		||||
 | 
			
		||||
<MCard Elevation="1" Rounded="false" Class="pa-2" Style="width:100%">
 | 
			
		||||
    <div class="mb-4">设备配置</div>
 | 
			
		||||
    <MRow Justify="JustifyTypes.Start" Align="AlignTypes.Center">
 | 
			
		||||
        <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.GroupSize) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.GroupSize />
 | 
			
		||||
         <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.UpdateRate) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.UpdateRate />
 | 
			
		||||
 | 
			
		||||
         <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.DeadBand) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.DeadBand />
 | 
			
		||||
         <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.KeepAliveInterval) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.KeepAliveInterval />
 | 
			
		||||
 | 
			
		||||
         <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.OPCUrl) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.OPCUrl />
 | 
			
		||||
         <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.UserName) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.UserName />
 | 
			
		||||
         <MTextField Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.Password) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.Password />
 | 
			
		||||
 | 
			
		||||
         <MCheckbox Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.ActiveSubscribe) Dense Outlined HideDetails="@("auto")" @bind-Value=@node.ActiveSubscribe />
 | 
			
		||||
         <MCheckbox Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.IsUseSecurity) Dense HideDetails="@("auto")" @bind-Value=@node.IsUseSecurity />
 | 
			
		||||
         <MCheckbox Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.CheckDomain) Dense HideDetails="@("auto")" @bind-Value=@node.CheckDomain />
 | 
			
		||||
         <MCheckbox Class="ma-1" Label=@node.DescriptionWithOutSugar(a=>a.LoadType) Dense HideDetails="@("auto")" @bind-Value=@node.LoadType />
 | 
			
		||||
 | 
			
		||||
         <MButton Class="ma-1" OnClick=@ConnectAsync Color="primary">
 | 
			
		||||
             连接
 | 
			
		||||
         </MButton>
 | 
			
		||||
 | 
			
		||||
         <MButton Class="ma-1" OnClick=@DisConnect Color="red">
 | 
			
		||||
             断开
 | 
			
		||||
         </MButton>
 | 
			
		||||
     </MRow>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 </MCard>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -1,43 +0,0 @@
 | 
			
		||||
<Project Sdk="Microsoft.NET.Sdk.Razor">
 | 
			
		||||
 | 
			
		||||
	<PropertyGroup>
 | 
			
		||||
		<DefineConstants>Plugin</DefineConstants>
 | 
			
		||||
	</PropertyGroup>
 | 
			
		||||
	
 | 
			
		||||
	<Target Name="PostBuild" AfterTargets="PostBuildEvent">
 | 
			
		||||
 | 
			
		||||
		<Exec Command=" set dir="$(SolutionDir)Web\ThingsGateway.Web.Entry\bin\$(Configuration)\$(TargetFramework)\Plugins\$(AssemblyName)"
 if not exist %25dir%25  md %25dir%25  
copy "$(TargetDir)*OPCUA*.dll"  %25dir%25
copy "$(TargetDir)*OPC.UA*.dll"  %25dir%25


" />
 | 
			
		||||
 | 
			
		||||
	</Target>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\DriverDebugUIBase.cs" Link="DebugPage\DriverDebugUIBase.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\DriverDebugUIPage.razor.cs" Link="DebugPage\DriverDebugUIPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\SerialSessionPage.razor.cs" Link="DebugPage\SerialSessionPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpClientPage.razor.cs" Link="DebugPage\TcpClientPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpServerPage.razor.cs" Link="DebugPage\TcpServerPage.razor.cs" />
 | 
			
		||||
		<Compile Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\UdpSessionPage.razor.cs" Link="DebugPage\UdpSessionPage.razor.cs" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\DriverDebugUIPage.razor" Link="DebugPage\DriverDebugUIPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\SerialSessionPage.razor" Link="DebugPage\SerialSessionPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpClientPage.razor" Link="DebugPage\TcpClientPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\TcpServerPage.razor" Link="DebugPage\TcpServerPage.razor" />
 | 
			
		||||
		<Content Include="..\..\Demo\ThingsGateway.Foundation.Demo.Rcl\Components\DebugPage\UdpSessionPage.razor" Link="DebugPage\UdpSessionPage.razor" />
 | 
			
		||||
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
	
 | 
			
		||||
	
 | 
			
		||||
	<ItemGroup>
 | 
			
		||||
		<PackageReference Include="OPCFoundation.NetStandard.Opc.Ua.Server" Version="1.4.372.56" />
 | 
			
		||||
		<ProjectReference Include="..\..\Foundation\ThingsGateway.Foundation.Adapter.OPCUA\ThingsGateway.Foundation.Adapter.OPCUA.csproj" />
 | 
			
		||||
		<ProjectReference Include="..\..\Web\ThingsGateway.Components\ThingsGateway.Components.csproj">
 | 
			
		||||
			<Private>false</Private>
 | 
			
		||||
			<CopyLocalSatelliteAssemblies>false</CopyLocalSatelliteAssemblies>
 | 
			
		||||
			<ExcludeAssets>runtime</ExcludeAssets>
 | 
			
		||||
		</ProjectReference>
 | 
			
		||||
	</ItemGroup>
 | 
			
		||||
 | 
			
		||||
	<PropertyGroup>
 | 
			
		||||
		<EnableDynamicLoading>true</EnableDynamicLoading>
 | 
			
		||||
	</PropertyGroup>
 | 
			
		||||
</Project>
 | 
			
		||||
@@ -1,35 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@using System.Net.Http
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Forms
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Routing
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web.Virtualization
 | 
			
		||||
@using Microsoft.JSInterop
 | 
			
		||||
@using BlazorComponent
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
@using Masa.Blazor.Presets
 | 
			
		||||
@using System.Net.Http.Json
 | 
			
		||||
@using System.IO;
 | 
			
		||||
@using System.Text.Json;
 | 
			
		||||
@using System.Reflection;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Components;
 | 
			
		||||
@using ThingsGateway.Admin.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Extension.String;
 | 
			
		||||
@using ThingsGateway.Admin.Application;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Core;
 | 
			
		||||
@using ThingsGateway.Gateway.Application;
 | 
			
		||||
@using ThingsGateway.Gateway.Core;
 | 
			
		||||
@@ -1,495 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using Furion;
 | 
			
		||||
 | 
			
		||||
using Mapster;
 | 
			
		||||
 | 
			
		||||
using Microsoft.Extensions.Logging;
 | 
			
		||||
 | 
			
		||||
using RabbitMQ.Client;
 | 
			
		||||
 | 
			
		||||
using System.Collections.Concurrent;
 | 
			
		||||
using System.Text;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Extension.ConcurrentQueue;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.Generic;
 | 
			
		||||
using ThingsGateway.Foundation.Extension.String;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.RabbitMQ;
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// RabbitMQ
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class RabbitMQClient : UpLoadBase
 | 
			
		||||
{
 | 
			
		||||
    private readonly RabbitMQClientProperty driverPropertys = new();
 | 
			
		||||
    private readonly RabbitMQClientVariableProperty variablePropertys = new();
 | 
			
		||||
    private ConcurrentQueue<DeviceData> _collectDeviceRunTimes = new();
 | 
			
		||||
    private ConcurrentQueue<VariableData> _collectVariableRunTimes = new();
 | 
			
		||||
    private IConnection _connection;
 | 
			
		||||
    private ConnectionFactory _connectionFactory;
 | 
			
		||||
    private GlobalDeviceData _globalDeviceData;
 | 
			
		||||
    private IModel _model;
 | 
			
		||||
    private RpcSingletonService _rpcCore;
 | 
			
		||||
    private List<DeviceVariableRunTime> _uploadVariables = new();
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Type DriverDebugUIType => null;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override UpDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override List<DeviceVariableRunTime> UploadVariables => _uploadVariables;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override VariablePropertyBase VariablePropertys => variablePropertys;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task AfterStopAsync()
 | 
			
		||||
    {
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override Task BeforStartAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        return Task.CompletedTask;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private TimerTick exDeviceTimerTick;
 | 
			
		||||
 | 
			
		||||
    private TimerTick exVariableTimerTick;
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override async Task ExecuteAsync(CancellationToken cancellationToken)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (_model == null)
 | 
			
		||||
            {
 | 
			
		||||
                try
 | 
			
		||||
                {
 | 
			
		||||
 | 
			
		||||
                    // 创建连接
 | 
			
		||||
                    _connection ??= _connectionFactory.CreateConnection();
 | 
			
		||||
                    // 创建通道
 | 
			
		||||
                    _model ??= _connection.CreateModel();
 | 
			
		||||
                    // 声明路由队列
 | 
			
		||||
                    if (driverPropertys.IsQueueDeclare)
 | 
			
		||||
                    {
 | 
			
		||||
                        _model?.QueueDeclare(driverPropertys.VariableQueueName, true, false, false);
 | 
			
		||||
                        _model?.QueueDeclare(driverPropertys.DeviceQueueName, true, false, false);
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
                catch (Exception ex)
 | 
			
		||||
                {
 | 
			
		||||
                    LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (!driverPropertys.IsInterval)
 | 
			
		||||
            {
 | 
			
		||||
                ////变化推送
 | 
			
		||||
                var varList = _collectVariableRunTimes.ToListWithDequeue();
 | 
			
		||||
                if (varList?.Count != 0)
 | 
			
		||||
                {
 | 
			
		||||
                    if (driverPropertys.IsList)
 | 
			
		||||
                    {
 | 
			
		||||
                        var listChunk = varList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                        foreach (var variables in listChunk)
 | 
			
		||||
                        {
 | 
			
		||||
                            try
 | 
			
		||||
                            {
 | 
			
		||||
                                if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                                {
 | 
			
		||||
                                    var data = variables.GetSciptListValue(driverPropertys.BigTextScriptVariableModel);
 | 
			
		||||
                                    // 设置消息持久化
 | 
			
		||||
                                    IBasicProperties properties = _model?.CreateBasicProperties();
 | 
			
		||||
                                    await Publish(driverPropertys.VariableQueueName, data, properties);
 | 
			
		||||
                                }
 | 
			
		||||
                                else
 | 
			
		||||
                                {
 | 
			
		||||
                                    break;
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                            catch (Exception ex)
 | 
			
		||||
                            {
 | 
			
		||||
                                LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                            }
 | 
			
		||||
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    else
 | 
			
		||||
                    {
 | 
			
		||||
                        foreach (var variable in varList)
 | 
			
		||||
                        {
 | 
			
		||||
                            try
 | 
			
		||||
                            {
 | 
			
		||||
                                if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                                {
 | 
			
		||||
                                    var data = variable.GetSciptListValue(driverPropertys.BigTextScriptVariableModel);
 | 
			
		||||
                                    // 设置消息持久化
 | 
			
		||||
                                    IBasicProperties properties = _model?.CreateBasicProperties();
 | 
			
		||||
                                    await Publish(driverPropertys.VariableQueueName, data, properties);
 | 
			
		||||
                                }
 | 
			
		||||
                                else
 | 
			
		||||
                                {
 | 
			
		||||
                                    break;
 | 
			
		||||
                                }
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
                            catch (Exception ex)
 | 
			
		||||
                            {
 | 
			
		||||
                                LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                if (exVariableTimerTick.IsTickHappen())
 | 
			
		||||
                {
 | 
			
		||||
                    try
 | 
			
		||||
                    {
 | 
			
		||||
                        var varList = _uploadVariables.Adapt<List<VariableData>>();
 | 
			
		||||
                        if (varList?.Count != 0)
 | 
			
		||||
                        {
 | 
			
		||||
                            if (driverPropertys.IsList)
 | 
			
		||||
                            {
 | 
			
		||||
                                var listChunk = varList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                                foreach (var variables in listChunk)
 | 
			
		||||
                                {
 | 
			
		||||
                                    try
 | 
			
		||||
                                    {
 | 
			
		||||
                                        if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                                        {
 | 
			
		||||
                                            var data = variables.GetSciptListValue(driverPropertys.BigTextScriptVariableModel);
 | 
			
		||||
                                            // 设置消息持久化
 | 
			
		||||
                                            IBasicProperties properties = _model?.CreateBasicProperties();
 | 
			
		||||
                                            await Publish(driverPropertys.VariableQueueName, data, properties);
 | 
			
		||||
                                        }
 | 
			
		||||
                                        else
 | 
			
		||||
                                        {
 | 
			
		||||
                                            break;
 | 
			
		||||
                                        }
 | 
			
		||||
                                    }
 | 
			
		||||
                                    catch (Exception ex)
 | 
			
		||||
                                    {
 | 
			
		||||
                                        LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                                    }
 | 
			
		||||
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                            else
 | 
			
		||||
                            {
 | 
			
		||||
                                foreach (var variable in varList)
 | 
			
		||||
                                {
 | 
			
		||||
                                    try
 | 
			
		||||
                                    {
 | 
			
		||||
                                        if (!cancellationToken.IsCancellationRequested)
 | 
			
		||||
                                        {
 | 
			
		||||
                                            var data = variable.GetSciptListValue(driverPropertys.BigTextScriptVariableModel);
 | 
			
		||||
                                            // 设置消息持久化
 | 
			
		||||
                                            IBasicProperties properties = _model?.CreateBasicProperties();
 | 
			
		||||
                                            await Publish(driverPropertys.VariableQueueName, data, properties);
 | 
			
		||||
                                        }
 | 
			
		||||
                                        else
 | 
			
		||||
                                        {
 | 
			
		||||
                                            break;
 | 
			
		||||
                                        }
 | 
			
		||||
 | 
			
		||||
                                    }
 | 
			
		||||
                                    catch (Exception ex)
 | 
			
		||||
                                    {
 | 
			
		||||
                                        LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                                    }
 | 
			
		||||
                                }
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    catch (Exception ex)
 | 
			
		||||
                    {
 | 
			
		||||
                        LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (!driverPropertys.IsInterval)
 | 
			
		||||
            {
 | 
			
		||||
                ////变化推送
 | 
			
		||||
                var devList = _collectDeviceRunTimes.ToListWithDequeue();
 | 
			
		||||
                if (devList?.Count != 0)
 | 
			
		||||
                {
 | 
			
		||||
                    if (driverPropertys.IsList)
 | 
			
		||||
                    {
 | 
			
		||||
                        var listChunk = devList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                        foreach (var devices in listChunk)
 | 
			
		||||
                        {
 | 
			
		||||
                            try
 | 
			
		||||
                            {
 | 
			
		||||
                                var data = devices.GetSciptListValue(driverPropertys.BigTextScriptDeviceModel);
 | 
			
		||||
                                // 设置消息持久化
 | 
			
		||||
                                IBasicProperties properties = _model?.CreateBasicProperties();
 | 
			
		||||
                                await Publish(driverPropertys.DeviceQueueName, data, properties);
 | 
			
		||||
                            }
 | 
			
		||||
                            catch (Exception ex)
 | 
			
		||||
                            {
 | 
			
		||||
                                LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                            }
 | 
			
		||||
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    else
 | 
			
		||||
                    {
 | 
			
		||||
                        foreach (var devices in devList)
 | 
			
		||||
                        {
 | 
			
		||||
                            try
 | 
			
		||||
                            {
 | 
			
		||||
                                var data = devices.GetSciptListValue(driverPropertys.BigTextScriptDeviceModel);
 | 
			
		||||
                                // 设置消息持久化
 | 
			
		||||
                                IBasicProperties properties = _model?.CreateBasicProperties();
 | 
			
		||||
                                await Publish(driverPropertys.DeviceQueueName, data, properties);
 | 
			
		||||
                            }
 | 
			
		||||
                            catch (Exception ex)
 | 
			
		||||
                            {
 | 
			
		||||
                                LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                if (exDeviceTimerTick.IsTickHappen())
 | 
			
		||||
                {
 | 
			
		||||
                    var devList = _collectDevice.Adapt<List<DeviceData>>();
 | 
			
		||||
                    if (devList?.Count != 0)
 | 
			
		||||
                    {
 | 
			
		||||
                        if (driverPropertys.IsList)
 | 
			
		||||
                        {
 | 
			
		||||
                            var listChunk = devList.ChunkTrivialBetter(driverPropertys.SplitSize);
 | 
			
		||||
                            foreach (var devices in listChunk)
 | 
			
		||||
                            {
 | 
			
		||||
                                try
 | 
			
		||||
                                {
 | 
			
		||||
                                    var data = devices.GetSciptListValue(driverPropertys.BigTextScriptDeviceModel);
 | 
			
		||||
                                    // 设置消息持久化
 | 
			
		||||
                                    IBasicProperties properties = _model?.CreateBasicProperties();
 | 
			
		||||
                                    await Publish(driverPropertys.DeviceQueueName, data, properties);
 | 
			
		||||
                                }
 | 
			
		||||
                                catch (Exception ex)
 | 
			
		||||
                                {
 | 
			
		||||
                                    LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                                }
 | 
			
		||||
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        else
 | 
			
		||||
                        {
 | 
			
		||||
                            foreach (var devices in devList)
 | 
			
		||||
                            {
 | 
			
		||||
                                try
 | 
			
		||||
                                {
 | 
			
		||||
                                    var data = devices.GetSciptListValue(driverPropertys.BigTextScriptDeviceModel);
 | 
			
		||||
                                    // 设置消息持久化
 | 
			
		||||
                                    IBasicProperties properties = _model?.CreateBasicProperties();
 | 
			
		||||
                                    await Publish(driverPropertys.DeviceQueueName, data, properties);
 | 
			
		||||
                                }
 | 
			
		||||
                                catch (Exception ex)
 | 
			
		||||
                                {
 | 
			
		||||
                                    LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (driverPropertys.CycleInterval > UploadDeviceThread.CycleInterval + 50)
 | 
			
		||||
        {
 | 
			
		||||
            try
 | 
			
		||||
            {
 | 
			
		||||
                await Task.Delay(driverPropertys.CycleInterval - UploadDeviceThread.CycleInterval, cancellationToken);
 | 
			
		||||
            }
 | 
			
		||||
            catch
 | 
			
		||||
            {
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public override bool IsConnected() => _connection?.IsOpen == true;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public override string ToString()
 | 
			
		||||
    {
 | 
			
		||||
        return $" {nameof(RabbitMQClient)} IP:{driverPropertys.IP} Port:{driverPropertys.Port}";
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Dispose(bool disposing)
 | 
			
		||||
    {
 | 
			
		||||
        _globalDeviceData?.AllVariables.ForEach(a => a.VariableValueChange -= VariableValueChange);
 | 
			
		||||
 | 
			
		||||
        _collectDevice?.ForEach(a =>
 | 
			
		||||
        {
 | 
			
		||||
            a.DeviceStatusChange -= DeviceStatusChange;
 | 
			
		||||
        });
 | 
			
		||||
        _model?.SafeDispose();
 | 
			
		||||
        _connection?.SafeDispose();
 | 
			
		||||
        _uploadVariables = null;
 | 
			
		||||
        _collectDeviceRunTimes.Clear();
 | 
			
		||||
        _collectVariableRunTimes.Clear();
 | 
			
		||||
        _collectDeviceRunTimes = null;
 | 
			
		||||
        _collectVariableRunTimes = null;
 | 
			
		||||
    }
 | 
			
		||||
    private List<CollectDeviceRunTime> _collectDevice;
 | 
			
		||||
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    protected override void Init(UploadDeviceRunTime device)
 | 
			
		||||
    {
 | 
			
		||||
        _connectionFactory = new ConnectionFactory
 | 
			
		||||
        {
 | 
			
		||||
            HostName = driverPropertys.IP,
 | 
			
		||||
            Port = driverPropertys.Port,
 | 
			
		||||
            UserName = driverPropertys.UserName,
 | 
			
		||||
            Password = driverPropertys.Password,
 | 
			
		||||
            VirtualHost = driverPropertys.VirtualHost,
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        _globalDeviceData = App.GetService<GlobalDeviceData>();
 | 
			
		||||
        _rpcCore = App.GetService<RpcSingletonService>();
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        var tags = _globalDeviceData.AllVariables.Where(a => a.VariablePropertys.ContainsKey(device.Id))
 | 
			
		||||
       .Where(b => b.VariablePropertys[device.Id].Any(c =>
 | 
			
		||||
       {
 | 
			
		||||
           if (c.PropertyName == nameof(variablePropertys.Enable))
 | 
			
		||||
           {
 | 
			
		||||
               return c.Value?.GetBoolValue() == true;
 | 
			
		||||
           }
 | 
			
		||||
           else
 | 
			
		||||
               return false;
 | 
			
		||||
       }))
 | 
			
		||||
       .ToList();
 | 
			
		||||
 | 
			
		||||
        _uploadVariables = tags;
 | 
			
		||||
 | 
			
		||||
        _collectDevice = _globalDeviceData.CollectDevices.Where(a => _uploadVariables.Select(b => b.DeviceId).Contains(a.Id)).ToList();
 | 
			
		||||
 | 
			
		||||
        _collectDevice.ForEach(a =>
 | 
			
		||||
        {
 | 
			
		||||
            a.DeviceStatusChange += DeviceStatusChange;
 | 
			
		||||
            DeviceStatusChange(a);
 | 
			
		||||
        });
 | 
			
		||||
        _uploadVariables.ForEach(a =>
 | 
			
		||||
        {
 | 
			
		||||
            a.VariableValueChange += VariableValueChange;
 | 
			
		||||
            VariableValueChange(a);
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
        if (driverPropertys.UploadInterval <= 1000) driverPropertys.UploadInterval = 1000;
 | 
			
		||||
        exVariableTimerTick = new(driverPropertys.UploadInterval);
 | 
			
		||||
        exDeviceTimerTick = new(driverPropertys.UploadInterval);
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private void DeviceStatusChange(CollectDeviceRunTime collectDeviceRunTime)
 | 
			
		||||
    {
 | 
			
		||||
        if (driverPropertys?.IsInterval != true)
 | 
			
		||||
            _collectDeviceRunTimes.Enqueue(collectDeviceRunTime.Adapt<DeviceData>());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task Publish(string queueName, string data, IBasicProperties properties)
 | 
			
		||||
    {
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            if (properties != null)
 | 
			
		||||
                properties.Persistent = true;
 | 
			
		||||
            if (_model != null)
 | 
			
		||||
                _model.BasicPublish(driverPropertys.ExchangeName, queueName, properties, Encoding.UTF8.GetBytes(data));
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                await CacheDb.AddCacheData(queueName, data, driverPropertys.CacheMaxCount);
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            //连接成功时补发缓存数据
 | 
			
		||||
            var cacheData = await CacheDb.GetCacheData(10);
 | 
			
		||||
            foreach (var item in cacheData)
 | 
			
		||||
            {
 | 
			
		||||
                try
 | 
			
		||||
                {
 | 
			
		||||
                    _model?.BasicPublish(driverPropertys.ExchangeName, item.Topic, properties, Encoding.UTF8.GetBytes(item.CacheStr));
 | 
			
		||||
                    LogMessage.Trace($"{FoundationConst.LogMessageHeader}主题:{item.Topic}{Environment.NewLine}负载:{item.CacheStr}");
 | 
			
		||||
 | 
			
		||||
                    await CacheDb.DeleteCacheData(item.Id);
 | 
			
		||||
                }
 | 
			
		||||
                catch
 | 
			
		||||
                {
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            LogMessage.Trace($"{FoundationConst.LogMessageHeader}主题:{queueName}{Environment.NewLine}负载:{data}");
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception ex)
 | 
			
		||||
        {
 | 
			
		||||
            LogMessage?.LogWarning(ex, ToString());
 | 
			
		||||
            await CacheDb.AddCacheData(queueName, data, driverPropertys.CacheMaxCount);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    private void VariableValueChange(DeviceVariableRunTime collectVariableRunTime)
 | 
			
		||||
    {
 | 
			
		||||
        if (driverPropertys?.IsInterval != true)
 | 
			
		||||
            _collectVariableRunTimes.Enqueue(collectVariableRunTime.Adapt<VariableData>());
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,68 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@page "/S7_1200"
 | 
			
		||||
 | 
			
		||||
@namespace ThingsGateway.Foundation.Demo
 | 
			
		||||
@using BlazorComponent;
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web;
 | 
			
		||||
@using Microsoft.JSInterop;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Foundation.Adapter.Siemens;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Extension;
 | 
			
		||||
@using ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
 | 
			
		||||
<TcpClientPage @ref=TcpClientPage></TcpClientPage>
 | 
			
		||||
 | 
			
		||||
<MCard Elevation="1" Rounded="false" Class="pa-2" Style="width:100%">
 | 
			
		||||
    <div class="mb-4">驱动配置</div>
 | 
			
		||||
    <MRow Justify="JustifyTypes.Start" Align="AlignTypes.Center">
 | 
			
		||||
        @if (_plc != null)
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.FrameTime)) Dense HideDetails="@("auto")" @bind-Value=@_plc.FrameTime></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.CacheTimeout)) Dense HideDetails="@("auto")" @bind-Value=@_plc.CacheTimeout></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.TimeOut)) Dense HideDetails="@("auto")" @bind-Value=@_plc.TimeOut></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Slot)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Slot></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Rack)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Rack></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MSelect Class="ma-1" Outlined Style="max-width:200px" @bind-Value="_plc.DataFormat" Label="@(_plc.DescriptionWithOutSugar(x => x.DataFormat))"
 | 
			
		||||
                     Items=@(typeof(DataFormat).GetEnumListWithOutSugar())
 | 
			
		||||
                     MenuProps="@(props => { props.Auto = true; props.OffsetY = true; })"
 | 
			
		||||
                     ItemText=@((u) =>u.Description)
 | 
			
		||||
                     ItemValue=@(u =>(DataFormat)u.Value)
 | 
			
		||||
                     HideDetails=@("auto") Height="30"
 | 
			
		||||
                  Dense>
 | 
			
		||||
        </MSelect>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
    </MRow>
 | 
			
		||||
</MCard>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
<DriverDebugUIPage @ref=driverDebugUIPage Sections="_sections">
 | 
			
		||||
 | 
			
		||||
</DriverDebugUIPage>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@code {
 | 
			
		||||
    private readonly List<(string Code, string Language)> _sections = new();
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -1,91 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Adapter.Siemens;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Demo;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
public partial class S7_1200DebugPage
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// SerialSessionPage
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private TcpClientPage TcpClientPage;
 | 
			
		||||
    private DriverDebugUIPage driverDebugUIPage;
 | 
			
		||||
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Siemens.SiemensS7PLC _plc;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="firstRender"></param>
 | 
			
		||||
    protected override void OnAfterRender(bool firstRender)
 | 
			
		||||
    {
 | 
			
		||||
        if (firstRender)
 | 
			
		||||
        {
 | 
			
		||||
            _sections.Add((
 | 
			
		||||
                """
 | 
			
		||||
                private static async Task ModbusClientAsync()
 | 
			
		||||
                {
 | 
			
		||||
                    //链路基础配置项
 | 
			
		||||
                    var config = new TouchSocketConfig();
 | 
			
		||||
                    config
 | 
			
		||||
                    .SetRemoteIPHost(new IPHost("127.0.0.1:502"))//TCP/UDP链路才需要
 | 
			
		||||
                
 | 
			
		||||
                var tcpClient1 = new TcpClient();//链路对象
 | 
			
		||||
                tcpClient1.Setup(config);
 | 
			
		||||
 | 
			
		||||
                    //创建协议对象,构造函数需要传入对应链路对象
 | 
			
		||||
                    SiemensS7PLC plc = new(tcpClient1,SiemensEnum.S1500)
 | 
			
		||||
                {
 | 
			
		||||
                    //协议配置
 | 
			
		||||
                    DataFormat = DataFormat.ABCD,
 | 
			
		||||
                    FrameTime = 0,
 | 
			
		||||
                    CacheTimeout = 1000,
 | 
			
		||||
                    ConnectTimeOut = 3000,
 | 
			
		||||
                    Station = 1,
 | 
			
		||||
                    TimeOut = 3000,
 | 
			
		||||
                    IsCheckMessageId = true
 | 
			
		||||
                };
 | 
			
		||||
 | 
			
		||||
                    #region 读写测试
 | 
			
		||||
                        var bytesResult = await plc.ReadAsync("400001", 20);
 | 
			
		||||
                        var int32sResult = await plc.ReadInt32Async("400001", 20);
 | 
			
		||||
 | 
			
		||||
                    #endregion
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
                
 | 
			
		||||
                """, "csharp"));
 | 
			
		||||
 | 
			
		||||
            if (TcpClientPage != null)
 | 
			
		||||
                TcpClientPage.LogAction = driverDebugUIPage.LogOut;
 | 
			
		||||
            _plc = new ThingsGateway.Foundation.Adapter.Siemens.SiemensS7PLC(TcpClientPage.GetTcpClient(), SiemensEnum.S1200);
 | 
			
		||||
            driverDebugUIPage.Plc = _plc;
 | 
			
		||||
 | 
			
		||||
            //初始化
 | 
			
		||||
            driverDebugUIPage.Address = "M100";
 | 
			
		||||
            int index = 0;
 | 
			
		||||
            driverDebugUIPage.DeviceVariableRunTimes.ForEach(a => a.VariableAddress = "M" + (100 + index++));
 | 
			
		||||
            TcpClientPage.Port = 102;
 | 
			
		||||
            TcpClientPage.StateHasChangedAsync();
 | 
			
		||||
 | 
			
		||||
            //载入配置
 | 
			
		||||
            StateHasChanged();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        base.OnAfterRender(firstRender);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,68 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@page "/S7_1500"
 | 
			
		||||
 | 
			
		||||
@namespace ThingsGateway.Foundation.Demo
 | 
			
		||||
@using BlazorComponent;
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web;
 | 
			
		||||
@using Microsoft.JSInterop;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Foundation.Adapter.Siemens;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Extension;
 | 
			
		||||
@using ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
 | 
			
		||||
<TcpClientPage @ref=TcpClientPage></TcpClientPage>
 | 
			
		||||
 | 
			
		||||
<MCard Elevation="1" Rounded="false" Class="pa-2" Style="width:100%">
 | 
			
		||||
    <div class="mb-4">驱动配置</div>
 | 
			
		||||
    <MRow Justify="JustifyTypes.Start" Align="AlignTypes.Center">
 | 
			
		||||
        @if (_plc != null)
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.FrameTime)) Dense HideDetails="@("auto")" @bind-Value=@_plc.FrameTime></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.CacheTimeout)) Dense HideDetails="@("auto")" @bind-Value=@_plc.CacheTimeout></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.TimeOut)) Dense HideDetails="@("auto")" @bind-Value=@_plc.TimeOut></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Slot)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Slot></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Rack)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Rack></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MSelect Class="ma-1" Outlined Style="max-width:200px" @bind-Value="_plc.DataFormat" Label="@(_plc.DescriptionWithOutSugar(x => x.DataFormat))"
 | 
			
		||||
                     Items=@(typeof(DataFormat).GetEnumListWithOutSugar())
 | 
			
		||||
                     MenuProps="@(props => { props.Auto = true; props.OffsetY = true; })"
 | 
			
		||||
                     ItemText=@((u) =>u.Description)
 | 
			
		||||
                     ItemValue=@(u =>(DataFormat)u.Value)
 | 
			
		||||
                     HideDetails=@("auto") Height="30"
 | 
			
		||||
                  Dense>
 | 
			
		||||
        </MSelect>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
    </MRow>
 | 
			
		||||
</MCard>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
<DriverDebugUIPage @ref=driverDebugUIPage Sections="_sections">
 | 
			
		||||
 | 
			
		||||
</DriverDebugUIPage>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@code {
 | 
			
		||||
    private readonly List<(string Code, string Language)> _sections = new();
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -1,92 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Adapter.Siemens;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Demo;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
public partial class S7_1500DebugPage
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// SerialSessionPage
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private TcpClientPage TcpClientPage;
 | 
			
		||||
    private DriverDebugUIPage driverDebugUIPage;
 | 
			
		||||
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Siemens.SiemensS7PLC _plc;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="firstRender"></param>
 | 
			
		||||
    protected override void OnAfterRender(bool firstRender)
 | 
			
		||||
    {
 | 
			
		||||
        if (firstRender)
 | 
			
		||||
        {
 | 
			
		||||
            _sections.Add((
 | 
			
		||||
                """
 | 
			
		||||
                private static async Task ModbusClientAsync()
 | 
			
		||||
                {
 | 
			
		||||
                    //链路基础配置项
 | 
			
		||||
                    var config = new TouchSocketConfig();
 | 
			
		||||
                    config
 | 
			
		||||
                    .SetRemoteIPHost(new IPHost("127.0.0.1:502"))//TCP/UDP链路才需要
 | 
			
		||||
                
 | 
			
		||||
                var tcpClient1 = new TcpClient();//链路对象
 | 
			
		||||
                tcpClient1.Setup(config);
 | 
			
		||||
 | 
			
		||||
                    //创建协议对象,构造函数需要传入对应链路对象
 | 
			
		||||
                    SiemensS7PLC plc = new(tcpClient1,SiemensEnum.S1500)
 | 
			
		||||
                {
 | 
			
		||||
                    //协议配置
 | 
			
		||||
                    DataFormat = DataFormat.ABCD,
 | 
			
		||||
                    FrameTime = 0,
 | 
			
		||||
                    CacheTimeout = 1000,
 | 
			
		||||
                    ConnectTimeOut = 3000,
 | 
			
		||||
                    Station = 1,
 | 
			
		||||
                    TimeOut = 3000,
 | 
			
		||||
                    IsCheckMessageId = true
 | 
			
		||||
                };
 | 
			
		||||
 | 
			
		||||
                    #region 读写测试
 | 
			
		||||
                        var bytesResult = await plc.ReadAsync("400001", 20);
 | 
			
		||||
                        var int32sResult = await plc.ReadInt32Async("400001", 20);
 | 
			
		||||
 | 
			
		||||
                    #endregion
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
                
 | 
			
		||||
                """, "csharp"));
 | 
			
		||||
 | 
			
		||||
            if (TcpClientPage != null)
 | 
			
		||||
                TcpClientPage.LogAction = driverDebugUIPage.LogOut;
 | 
			
		||||
            _plc = new ThingsGateway.Foundation.Adapter.Siemens.SiemensS7PLC(TcpClientPage.GetTcpClient(), SiemensEnum.S1500);
 | 
			
		||||
            driverDebugUIPage.Plc = _plc;
 | 
			
		||||
 | 
			
		||||
            //初始化
 | 
			
		||||
            driverDebugUIPage.Address = "M100";
 | 
			
		||||
            int index = 0;
 | 
			
		||||
            driverDebugUIPage.DeviceVariableRunTimes.ForEach(a => a.VariableAddress = "M" + (100 + index++));
 | 
			
		||||
            TcpClientPage.Port = 102;
 | 
			
		||||
            TcpClientPage.StateHasChangedAsync();
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
            //载入配置
 | 
			
		||||
            StateHasChanged();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        base.OnAfterRender(firstRender);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,68 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@page "/S7_200"
 | 
			
		||||
 | 
			
		||||
@namespace ThingsGateway.Foundation.Demo
 | 
			
		||||
@using BlazorComponent;
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web;
 | 
			
		||||
@using Microsoft.JSInterop;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Foundation.Adapter.Siemens;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Extension;
 | 
			
		||||
@using ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
 | 
			
		||||
<TcpClientPage @ref=TcpClientPage></TcpClientPage>
 | 
			
		||||
 | 
			
		||||
<MCard Elevation="1" Rounded="false" Class="pa-2" Style="width:100%">
 | 
			
		||||
    <div class="mb-4">驱动配置</div>
 | 
			
		||||
    <MRow Justify="JustifyTypes.Start" Align="AlignTypes.Center">
 | 
			
		||||
        @if (_plc != null)
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.FrameTime)) Dense HideDetails="@("auto")" @bind-Value=@_plc.FrameTime></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.CacheTimeout)) Dense HideDetails="@("auto")" @bind-Value=@_plc.CacheTimeout></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.TimeOut)) Dense HideDetails="@("auto")" @bind-Value=@_plc.TimeOut></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Slot)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Slot></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Rack)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Rack></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MSelect Class="ma-1" Outlined Style="max-width:200px" @bind-Value="_plc.DataFormat" Label="@(_plc.DescriptionWithOutSugar(x => x.DataFormat))"
 | 
			
		||||
                     Items=@(typeof(DataFormat).GetEnumListWithOutSugar())
 | 
			
		||||
                     MenuProps="@(props => { props.Auto = true; props.OffsetY = true; })"
 | 
			
		||||
                     ItemText=@((u) =>u.Description)
 | 
			
		||||
                     ItemValue=@(u =>(DataFormat)u.Value)
 | 
			
		||||
                     HideDetails=@("auto") Height="30"
 | 
			
		||||
                  Dense>
 | 
			
		||||
        </MSelect>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
    </MRow>
 | 
			
		||||
</MCard>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
<DriverDebugUIPage @ref=driverDebugUIPage Sections="_sections">
 | 
			
		||||
 | 
			
		||||
</DriverDebugUIPage>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@code {
 | 
			
		||||
    private readonly List<(string Code, string Language)> _sections = new();
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -1,92 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Adapter.Siemens;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Demo;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
public partial class S7_200DebugPage
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// SerialSessionPage
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private TcpClientPage TcpClientPage;
 | 
			
		||||
    private DriverDebugUIPage driverDebugUIPage;
 | 
			
		||||
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Siemens.SiemensS7PLC _plc;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="firstRender"></param>
 | 
			
		||||
    protected override void OnAfterRender(bool firstRender)
 | 
			
		||||
    {
 | 
			
		||||
        if (firstRender)
 | 
			
		||||
        {
 | 
			
		||||
            _sections.Add((
 | 
			
		||||
                """
 | 
			
		||||
                private static async Task ModbusClientAsync()
 | 
			
		||||
                {
 | 
			
		||||
                    //链路基础配置项
 | 
			
		||||
                    var config = new TouchSocketConfig();
 | 
			
		||||
                    config
 | 
			
		||||
                    .SetRemoteIPHost(new IPHost("127.0.0.1:502"))//TCP/UDP链路才需要
 | 
			
		||||
                
 | 
			
		||||
                var tcpClient1 = new TcpClient();//链路对象
 | 
			
		||||
                tcpClient1.Setup(config);
 | 
			
		||||
 | 
			
		||||
                    //创建协议对象,构造函数需要传入对应链路对象
 | 
			
		||||
                    SiemensS7PLC plc = new(tcpClient1,SiemensEnum.S1500)
 | 
			
		||||
                {
 | 
			
		||||
                    //协议配置
 | 
			
		||||
                    DataFormat = DataFormat.ABCD,
 | 
			
		||||
                    FrameTime = 0,
 | 
			
		||||
                    CacheTimeout = 1000,
 | 
			
		||||
                    ConnectTimeOut = 3000,
 | 
			
		||||
                    Station = 1,
 | 
			
		||||
                    TimeOut = 3000,
 | 
			
		||||
                    IsCheckMessageId = true
 | 
			
		||||
                };
 | 
			
		||||
 | 
			
		||||
                    #region 读写测试
 | 
			
		||||
                        var bytesResult = await plc.ReadAsync("400001", 20);
 | 
			
		||||
                        var int32sResult = await plc.ReadInt32Async("400001", 20);
 | 
			
		||||
 | 
			
		||||
                    #endregion
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
                
 | 
			
		||||
                """, "csharp"));
 | 
			
		||||
 | 
			
		||||
            if (TcpClientPage != null)
 | 
			
		||||
                TcpClientPage.LogAction = driverDebugUIPage.LogOut;
 | 
			
		||||
            _plc = new ThingsGateway.Foundation.Adapter.Siemens.SiemensS7PLC(TcpClientPage.GetTcpClient(), SiemensEnum.S200);
 | 
			
		||||
            driverDebugUIPage.Plc = _plc;
 | 
			
		||||
 | 
			
		||||
            //初始化
 | 
			
		||||
            driverDebugUIPage.Address = "M100";
 | 
			
		||||
            int index = 0;
 | 
			
		||||
            driverDebugUIPage.DeviceVariableRunTimes.ForEach(a => a.VariableAddress = "M" + (100 + index++));
 | 
			
		||||
            TcpClientPage.Port = 102;
 | 
			
		||||
            TcpClientPage.StateHasChangedAsync();
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
            //载入配置
 | 
			
		||||
            StateHasChanged();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        base.OnAfterRender(firstRender);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,68 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@page "/S7_200SMART"
 | 
			
		||||
 | 
			
		||||
@namespace ThingsGateway.Foundation.Demo
 | 
			
		||||
@using BlazorComponent;
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web;
 | 
			
		||||
@using Microsoft.JSInterop;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Foundation.Adapter.Siemens;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Extension;
 | 
			
		||||
@using ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
 | 
			
		||||
<TcpClientPage @ref=TcpClientPage></TcpClientPage>
 | 
			
		||||
 | 
			
		||||
<MCard Elevation="1" Rounded="false" Class="pa-2" Style="width:100%">
 | 
			
		||||
    <div class="mb-4">驱动配置</div>
 | 
			
		||||
    <MRow Justify="JustifyTypes.Start" Align="AlignTypes.Center">
 | 
			
		||||
        @if (_plc != null)
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.FrameTime)) Dense HideDetails="@("auto")" @bind-Value=@_plc.FrameTime></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.CacheTimeout)) Dense HideDetails="@("auto")" @bind-Value=@_plc.CacheTimeout></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.TimeOut)) Dense HideDetails="@("auto")" @bind-Value=@_plc.TimeOut></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Slot)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Slot></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Rack)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Rack></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MSelect Class="ma-1" Outlined Style="max-width:200px" @bind-Value="_plc.DataFormat" Label="@(_plc.DescriptionWithOutSugar(x => x.DataFormat))"
 | 
			
		||||
                     Items=@(typeof(DataFormat).GetEnumListWithOutSugar())
 | 
			
		||||
                     MenuProps="@(props => { props.Auto = true; props.OffsetY = true; })"
 | 
			
		||||
                     ItemText=@((u) =>u.Description)
 | 
			
		||||
                     ItemValue=@(u =>(DataFormat)u.Value)
 | 
			
		||||
                     HideDetails=@("auto") Height="30"
 | 
			
		||||
                  Dense>
 | 
			
		||||
        </MSelect>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
    </MRow>
 | 
			
		||||
</MCard>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
<DriverDebugUIPage @ref=driverDebugUIPage Sections="_sections">
 | 
			
		||||
 | 
			
		||||
</DriverDebugUIPage>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@code {
 | 
			
		||||
    private readonly List<(string Code, string Language)> _sections = new();
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -1,92 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Adapter.Siemens;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Demo;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
public partial class S7_200SMARTDebugPage
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// SerialSessionPage
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private TcpClientPage TcpClientPage;
 | 
			
		||||
    private DriverDebugUIPage driverDebugUIPage;
 | 
			
		||||
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Siemens.SiemensS7PLC _plc;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="firstRender"></param>
 | 
			
		||||
    protected override void OnAfterRender(bool firstRender)
 | 
			
		||||
    {
 | 
			
		||||
        if (firstRender)
 | 
			
		||||
        {
 | 
			
		||||
            _sections.Add((
 | 
			
		||||
                """
 | 
			
		||||
                private static async Task ModbusClientAsync()
 | 
			
		||||
                {
 | 
			
		||||
                    //链路基础配置项
 | 
			
		||||
                    var config = new TouchSocketConfig();
 | 
			
		||||
                    config
 | 
			
		||||
                    .SetRemoteIPHost(new IPHost("127.0.0.1:502"))//TCP/UDP链路才需要
 | 
			
		||||
                
 | 
			
		||||
                var tcpClient1 = new TcpClient();//链路对象
 | 
			
		||||
                tcpClient1.Setup(config);
 | 
			
		||||
 | 
			
		||||
                    //创建协议对象,构造函数需要传入对应链路对象
 | 
			
		||||
                    SiemensS7PLC plc = new(tcpClient1,SiemensEnum.S1500)
 | 
			
		||||
                {
 | 
			
		||||
                    //协议配置
 | 
			
		||||
                    DataFormat = DataFormat.ABCD,
 | 
			
		||||
                    FrameTime = 0,
 | 
			
		||||
                    CacheTimeout = 1000,
 | 
			
		||||
                    ConnectTimeOut = 3000,
 | 
			
		||||
                    Station = 1,
 | 
			
		||||
                    TimeOut = 3000,
 | 
			
		||||
                    IsCheckMessageId = true
 | 
			
		||||
                };
 | 
			
		||||
 | 
			
		||||
                    #region 读写测试
 | 
			
		||||
                        var bytesResult = await plc.ReadAsync("400001", 20);
 | 
			
		||||
                        var int32sResult = await plc.ReadInt32Async("400001", 20);
 | 
			
		||||
 | 
			
		||||
                    #endregion
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
                
 | 
			
		||||
                """, "csharp"));
 | 
			
		||||
 | 
			
		||||
            if (TcpClientPage != null)
 | 
			
		||||
                TcpClientPage.LogAction = driverDebugUIPage.LogOut;
 | 
			
		||||
            _plc = new ThingsGateway.Foundation.Adapter.Siemens.SiemensS7PLC(TcpClientPage.GetTcpClient(), SiemensEnum.S200Smart);
 | 
			
		||||
            driverDebugUIPage.Plc = _plc;
 | 
			
		||||
 | 
			
		||||
            //初始化
 | 
			
		||||
            driverDebugUIPage.Address = "M100";
 | 
			
		||||
            int index = 0;
 | 
			
		||||
            driverDebugUIPage.DeviceVariableRunTimes.ForEach(a => a.VariableAddress = "M" + (100 + index++));
 | 
			
		||||
            TcpClientPage.Port = 102;
 | 
			
		||||
            TcpClientPage.StateHasChangedAsync();
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
            //载入配置
 | 
			
		||||
            StateHasChanged();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        base.OnAfterRender(firstRender);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,68 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@page "/S7_300"
 | 
			
		||||
 | 
			
		||||
@namespace ThingsGateway.Foundation.Demo
 | 
			
		||||
@using BlazorComponent;
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web;
 | 
			
		||||
@using Microsoft.JSInterop;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Foundation.Adapter.Siemens;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Extension;
 | 
			
		||||
@using ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
 | 
			
		||||
<TcpClientPage @ref=TcpClientPage></TcpClientPage>
 | 
			
		||||
 | 
			
		||||
<MCard Elevation="1" Rounded="false" Class="pa-2" Style="width:100%">
 | 
			
		||||
    <div class="mb-4">驱动配置</div>
 | 
			
		||||
    <MRow Justify="JustifyTypes.Start" Align="AlignTypes.Center">
 | 
			
		||||
        @if (_plc != null)
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.FrameTime)) Dense HideDetails="@("auto")" @bind-Value=@_plc.FrameTime></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.CacheTimeout)) Dense HideDetails="@("auto")" @bind-Value=@_plc.CacheTimeout></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.TimeOut)) Dense HideDetails="@("auto")" @bind-Value=@_plc.TimeOut></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Slot)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Slot></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Rack)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Rack></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MSelect Class="ma-1" Outlined Style="max-width:200px" @bind-Value="_plc.DataFormat" Label="@(_plc.DescriptionWithOutSugar(x => x.DataFormat))"
 | 
			
		||||
                     Items=@(typeof(DataFormat).GetEnumListWithOutSugar())
 | 
			
		||||
                     MenuProps="@(props => { props.Auto = true; props.OffsetY = true; })"
 | 
			
		||||
                     ItemText=@((u) =>u.Description)
 | 
			
		||||
                     ItemValue=@(u =>(DataFormat)u.Value)
 | 
			
		||||
                     HideDetails=@("auto") Height="30"
 | 
			
		||||
                  Dense>
 | 
			
		||||
        </MSelect>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
    </MRow>
 | 
			
		||||
</MCard>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
<DriverDebugUIPage @ref=driverDebugUIPage Sections="_sections">
 | 
			
		||||
 | 
			
		||||
</DriverDebugUIPage>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@code {
 | 
			
		||||
    private readonly List<(string Code, string Language)> _sections = new();
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -1,68 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@page "/S7_400"
 | 
			
		||||
 | 
			
		||||
@namespace ThingsGateway.Foundation.Demo
 | 
			
		||||
@using BlazorComponent;
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web;
 | 
			
		||||
@using Microsoft.JSInterop;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Foundation.Adapter.Siemens;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Extension;
 | 
			
		||||
@using ThingsGateway.Foundation.Serial;
 | 
			
		||||
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
 | 
			
		||||
<TcpClientPage @ref=TcpClientPage></TcpClientPage>
 | 
			
		||||
 | 
			
		||||
<MCard Elevation="1" Rounded="false" Class="pa-2" Style="width:100%">
 | 
			
		||||
    <div class="mb-4">驱动配置</div>
 | 
			
		||||
    <MRow Justify="JustifyTypes.Start" Align="AlignTypes.Center">
 | 
			
		||||
        @if (_plc != null)
 | 
			
		||||
        {
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.FrameTime)) Dense HideDetails="@("auto")" @bind-Value=@_plc.FrameTime></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.CacheTimeout)) Dense HideDetails="@("auto")" @bind-Value=@_plc.CacheTimeout></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.TimeOut)) Dense HideDetails="@("auto")" @bind-Value=@_plc.TimeOut></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Slot)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Slot></MTextField>
 | 
			
		||||
            <MTextField Class="ma-1" Outlined Style="max-width:100px" Label=@(_plc.DescriptionWithOutSugar(x => x.Rack)) Dense HideDetails="@("auto")" @bind-Value=@_plc.Rack></MTextField>
 | 
			
		||||
 | 
			
		||||
            <MSelect Class="ma-1" Outlined Style="max-width:200px" @bind-Value="_plc.DataFormat" Label="@(_plc.DescriptionWithOutSugar(x => x.DataFormat))"
 | 
			
		||||
                     Items=@(typeof(DataFormat).GetEnumListWithOutSugar())
 | 
			
		||||
                     MenuProps="@(props => { props.Auto = true; props.OffsetY = true; })"
 | 
			
		||||
                     ItemText=@((u) =>u.Description)
 | 
			
		||||
                     ItemValue=@(u =>(DataFormat)u.Value)
 | 
			
		||||
                     HideDetails=@("auto") Height="30"
 | 
			
		||||
                  Dense>
 | 
			
		||||
        </MSelect>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        }
 | 
			
		||||
    </MRow>
 | 
			
		||||
</MCard>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
<DriverDebugUIPage @ref=driverDebugUIPage Sections="_sections">
 | 
			
		||||
 | 
			
		||||
</DriverDebugUIPage>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@code {
 | 
			
		||||
    private readonly List<(string Code, string Language)> _sections = new();
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -1,92 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Adapter.Siemens;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Foundation.Demo;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/// <inheritdoc/>
 | 
			
		||||
public partial class S7_400DebugPage
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// SerialSessionPage
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    private TcpClientPage TcpClientPage;
 | 
			
		||||
    private DriverDebugUIPage driverDebugUIPage;
 | 
			
		||||
 | 
			
		||||
    private ThingsGateway.Foundation.Adapter.Siemens.SiemensS7PLC _plc;
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="firstRender"></param>
 | 
			
		||||
    protected override void OnAfterRender(bool firstRender)
 | 
			
		||||
    {
 | 
			
		||||
        if (firstRender)
 | 
			
		||||
        {
 | 
			
		||||
            _sections.Add((
 | 
			
		||||
                """
 | 
			
		||||
                private static async Task ModbusClientAsync()
 | 
			
		||||
                {
 | 
			
		||||
                    //链路基础配置项
 | 
			
		||||
                    var config = new TouchSocketConfig();
 | 
			
		||||
                    config
 | 
			
		||||
                    .SetRemoteIPHost(new IPHost("127.0.0.1:502"))//TCP/UDP链路才需要
 | 
			
		||||
                
 | 
			
		||||
                var tcpClient1 = new TcpClient();//链路对象
 | 
			
		||||
                tcpClient1.Setup(config);
 | 
			
		||||
 | 
			
		||||
                    //创建协议对象,构造函数需要传入对应链路对象
 | 
			
		||||
                    SiemensS7PLC plc = new(tcpClient1,SiemensEnum.S1500)
 | 
			
		||||
                {
 | 
			
		||||
                    //协议配置
 | 
			
		||||
                    DataFormat = DataFormat.ABCD,
 | 
			
		||||
                    FrameTime = 0,
 | 
			
		||||
                    CacheTimeout = 1000,
 | 
			
		||||
                    ConnectTimeOut = 3000,
 | 
			
		||||
                    Station = 1,
 | 
			
		||||
                    TimeOut = 3000,
 | 
			
		||||
                    IsCheckMessageId = true
 | 
			
		||||
                };
 | 
			
		||||
 | 
			
		||||
                    #region 读写测试
 | 
			
		||||
                        var bytesResult = await plc.ReadAsync("400001", 20);
 | 
			
		||||
                        var int32sResult = await plc.ReadInt32Async("400001", 20);
 | 
			
		||||
 | 
			
		||||
                    #endregion
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
                
 | 
			
		||||
                """, "csharp"));
 | 
			
		||||
 | 
			
		||||
            if (TcpClientPage != null)
 | 
			
		||||
                TcpClientPage.LogAction = driverDebugUIPage.LogOut;
 | 
			
		||||
            _plc = new ThingsGateway.Foundation.Adapter.Siemens.SiemensS7PLC(TcpClientPage.GetTcpClient(), SiemensEnum.S400);
 | 
			
		||||
            driverDebugUIPage.Plc = _plc;
 | 
			
		||||
 | 
			
		||||
            //初始化
 | 
			
		||||
            driverDebugUIPage.Address = "M100";
 | 
			
		||||
            int index = 0;
 | 
			
		||||
            driverDebugUIPage.DeviceVariableRunTimes.ForEach(a => a.VariableAddress = "M" + (100 + index++));
 | 
			
		||||
            TcpClientPage.Port = 102;
 | 
			
		||||
            TcpClientPage.StateHasChangedAsync();
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
            //载入配置
 | 
			
		||||
            StateHasChanged();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        base.OnAfterRender(firstRender);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,54 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Siemens
 | 
			
		||||
{
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public class S7_1200 : Siemens
 | 
			
		||||
    {
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        public override Type DriverDebugUIType => typeof(S7_1200DebugPage);
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
        {
 | 
			
		||||
            if (client == null)
 | 
			
		||||
            {
 | 
			
		||||
                FoundataionConfig.SetRemoteIPHost(new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}"))
 | 
			
		||||
                    ;
 | 
			
		||||
                client = new TcpClient();
 | 
			
		||||
                ((TcpClient)client).Setup(FoundataionConfig);
 | 
			
		||||
            }
 | 
			
		||||
            //载入配置
 | 
			
		||||
            _plc = new((TcpClient)client, SiemensEnum.S1200)
 | 
			
		||||
            {
 | 
			
		||||
                DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
                ConnectTimeOut = driverPropertys.ConnectTimeOut,
 | 
			
		||||
                TimeOut = driverPropertys.TimeOut
 | 
			
		||||
            };
 | 
			
		||||
            if (driverPropertys.LocalTSAP != 0)
 | 
			
		||||
            {
 | 
			
		||||
                _plc.LocalTSAP = driverPropertys.LocalTSAP;
 | 
			
		||||
            }
 | 
			
		||||
            if (driverPropertys.DestTSAP != 0)
 | 
			
		||||
            {
 | 
			
		||||
                _plc.DestTSAP = driverPropertys.DestTSAP;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,53 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Siemens
 | 
			
		||||
{
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public class S7_1500 : Siemens
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override Type DriverDebugUIType => typeof(S7_1500DebugPage);
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
        {
 | 
			
		||||
            if (client == null)
 | 
			
		||||
            {
 | 
			
		||||
                FoundataionConfig.SetRemoteIPHost(new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}"))
 | 
			
		||||
                    ;
 | 
			
		||||
                client = new TcpClient();
 | 
			
		||||
                ((TcpClient)client).Setup(FoundataionConfig);
 | 
			
		||||
            }
 | 
			
		||||
            //载入配置
 | 
			
		||||
            _plc = new((TcpClient)client, SiemensEnum.S1500)
 | 
			
		||||
            {
 | 
			
		||||
                DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
                ConnectTimeOut = driverPropertys.ConnectTimeOut,
 | 
			
		||||
                TimeOut = driverPropertys.TimeOut
 | 
			
		||||
            };
 | 
			
		||||
            if (driverPropertys.LocalTSAP != 0)
 | 
			
		||||
            {
 | 
			
		||||
                _plc.LocalTSAP = driverPropertys.LocalTSAP;
 | 
			
		||||
            }
 | 
			
		||||
            if (driverPropertys.DestTSAP != 0)
 | 
			
		||||
            {
 | 
			
		||||
                _plc.DestTSAP = driverPropertys.DestTSAP;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,53 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Siemens
 | 
			
		||||
{
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public class S7_200 : Siemens
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override Type DriverDebugUIType => typeof(S7_200DebugPage);
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
        {
 | 
			
		||||
            if (client == null)
 | 
			
		||||
            {
 | 
			
		||||
                FoundataionConfig.SetRemoteIPHost(new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}"))
 | 
			
		||||
                    ;
 | 
			
		||||
                client = new TcpClient();
 | 
			
		||||
                ((TcpClient)client).Setup(FoundataionConfig);
 | 
			
		||||
            }
 | 
			
		||||
            //载入配置
 | 
			
		||||
            _plc = new((TcpClient)client, SiemensEnum.S200)
 | 
			
		||||
            {
 | 
			
		||||
                DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
                ConnectTimeOut = driverPropertys.ConnectTimeOut,
 | 
			
		||||
                TimeOut = driverPropertys.TimeOut
 | 
			
		||||
            };
 | 
			
		||||
            if (driverPropertys.LocalTSAP != 0)
 | 
			
		||||
            {
 | 
			
		||||
                _plc.LocalTSAP = driverPropertys.LocalTSAP;
 | 
			
		||||
            }
 | 
			
		||||
            if (driverPropertys.DestTSAP != 0)
 | 
			
		||||
            {
 | 
			
		||||
                _plc.DestTSAP = driverPropertys.DestTSAP;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,52 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Siemens
 | 
			
		||||
{
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public class S7_200SMART : Siemens
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override Type DriverDebugUIType => typeof(S7_200SMARTDebugPage);
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
        {
 | 
			
		||||
            if (client == null)
 | 
			
		||||
            {
 | 
			
		||||
                FoundataionConfig.SetRemoteIPHost(new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}"))
 | 
			
		||||
                    ;
 | 
			
		||||
                client = new TcpClient();
 | 
			
		||||
                ((TcpClient)client).Setup(FoundataionConfig);
 | 
			
		||||
            }
 | 
			
		||||
            //载入配置
 | 
			
		||||
            _plc = new((TcpClient)client, SiemensEnum.S200Smart)
 | 
			
		||||
            {
 | 
			
		||||
                DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
                ConnectTimeOut = driverPropertys.ConnectTimeOut,
 | 
			
		||||
                TimeOut = driverPropertys.TimeOut
 | 
			
		||||
            };
 | 
			
		||||
            if (driverPropertys.LocalTSAP != 0)
 | 
			
		||||
            {
 | 
			
		||||
                _plc.LocalTSAP = driverPropertys.LocalTSAP;
 | 
			
		||||
            }
 | 
			
		||||
            if (driverPropertys.DestTSAP != 0)
 | 
			
		||||
            {
 | 
			
		||||
                _plc.DestTSAP = driverPropertys.DestTSAP;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,53 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Siemens
 | 
			
		||||
{
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public class S7_300 : Siemens
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override Type DriverDebugUIType => typeof(S7_300DebugPage);
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
        {
 | 
			
		||||
            if (client == null)
 | 
			
		||||
            {
 | 
			
		||||
                FoundataionConfig.SetRemoteIPHost(new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}"))
 | 
			
		||||
                    ;
 | 
			
		||||
                client = new TcpClient();
 | 
			
		||||
                ((TcpClient)client).Setup(FoundataionConfig);
 | 
			
		||||
            }
 | 
			
		||||
            //载入配置
 | 
			
		||||
            _plc = new((TcpClient)client, SiemensEnum.S300)
 | 
			
		||||
            {
 | 
			
		||||
                DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
                ConnectTimeOut = driverPropertys.ConnectTimeOut,
 | 
			
		||||
                TimeOut = driverPropertys.TimeOut
 | 
			
		||||
            };
 | 
			
		||||
            if (driverPropertys.LocalTSAP != 0)
 | 
			
		||||
            {
 | 
			
		||||
                _plc.LocalTSAP = driverPropertys.LocalTSAP;
 | 
			
		||||
            }
 | 
			
		||||
            if (driverPropertys.DestTSAP != 0)
 | 
			
		||||
            {
 | 
			
		||||
                _plc.DestTSAP = driverPropertys.DestTSAP;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,53 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Plugin.Siemens
 | 
			
		||||
{
 | 
			
		||||
    /// <inheritdoc/>
 | 
			
		||||
    public class S7_400 : Siemens
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override Type DriverDebugUIType => typeof(S7_400DebugPage);
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        public override CollectDriverPropertyBase DriverPropertys => driverPropertys;
 | 
			
		||||
 | 
			
		||||
        /// <inheritdoc/>
 | 
			
		||||
        protected override void Init(CollectDeviceRunTime device, object client = null)
 | 
			
		||||
        {
 | 
			
		||||
            if (client == null)
 | 
			
		||||
            {
 | 
			
		||||
                FoundataionConfig.SetRemoteIPHost(new IPHost($"{driverPropertys.IP}:{driverPropertys.Port}"))
 | 
			
		||||
                    ;
 | 
			
		||||
                client = new TcpClient();
 | 
			
		||||
                ((TcpClient)client).Setup(FoundataionConfig);
 | 
			
		||||
            }
 | 
			
		||||
            //载入配置
 | 
			
		||||
            _plc = new((TcpClient)client, SiemensEnum.S400)
 | 
			
		||||
            {
 | 
			
		||||
                DataFormat = driverPropertys.DataFormat,
 | 
			
		||||
                ConnectTimeOut = driverPropertys.ConnectTimeOut,
 | 
			
		||||
                TimeOut = driverPropertys.TimeOut
 | 
			
		||||
            };
 | 
			
		||||
            if (driverPropertys.LocalTSAP != 0)
 | 
			
		||||
            {
 | 
			
		||||
                _plc.LocalTSAP = driverPropertys.LocalTSAP;
 | 
			
		||||
            }
 | 
			
		||||
            if (driverPropertys.DestTSAP != 0)
 | 
			
		||||
            {
 | 
			
		||||
                _plc.DestTSAP = driverPropertys.DestTSAP;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,35 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@using System.Net.Http
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Forms
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Routing
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web
 | 
			
		||||
@using Microsoft.AspNetCore.Components.Web.Virtualization
 | 
			
		||||
@using Microsoft.JSInterop
 | 
			
		||||
@using BlazorComponent
 | 
			
		||||
@using Masa.Blazor
 | 
			
		||||
@using Masa.Blazor.Presets
 | 
			
		||||
@using System.Net.Http.Json
 | 
			
		||||
@using System.IO;
 | 
			
		||||
@using System.Text.Json;
 | 
			
		||||
@using System.Reflection;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Components;
 | 
			
		||||
@using ThingsGateway.Admin.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Core;
 | 
			
		||||
@using ThingsGateway.Foundation.Extension.String;
 | 
			
		||||
@using ThingsGateway.Admin.Application;
 | 
			
		||||
 | 
			
		||||
@using ThingsGateway.Core;
 | 
			
		||||
@using ThingsGateway.Gateway.Application;
 | 
			
		||||
@using ThingsGateway.Gateway.Core;
 | 
			
		||||
@@ -3,48 +3,40 @@ Microsoft Visual Studio Solution File, Format Version 12.00
 | 
			
		||||
# Visual Studio Version 17
 | 
			
		||||
VisualStudioVersion = 17.6.33927.249
 | 
			
		||||
MinimumVisualStudioVersion = 10.0.40219.1
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation", "Foundation\ThingsGateway.Foundation\ThingsGateway.Foundation.csproj", "{2070116F-7A1B-47E2-A2F2-7BEC29ECE891}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Web", "Web", "{9EB46BB6-D4EA-4B06-95EE-6C971E653030}"
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "admin", "admin", "{4E66C22C-0636-4949-BF6A-9E3BBE1550BA}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		Web\Directory.Build.props = Web\Directory.Build.props
 | 
			
		||||
		admin\Directory.Build.props = admin\Directory.Build.props
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "script", "script", "{4A64518E-C072-4607-BBF7-7D392CEC9D58}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Components", "admin\ThingsGateway.Components\ThingsGateway.Components.csproj", "{0A891D8E-23B3-46AD-8D30-565EE5004F93}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Core", "admin\ThingsGateway.Core\ThingsGateway.Core.csproj", "{A712EAEE-94F2-4F01-8C1C-2EC802280DD7}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.Core", "admin\ThingsGateway.Admin.Core\ThingsGateway.Admin.Core.csproj", "{5DA3D2BD-6768-4479-B52F-49E022EFF310}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.Blazor", "admin\ThingsGateway.Admin.Blazor\ThingsGateway.Admin.Blazor.csproj", "{8DD5DF98-7FDE-4B49-8661-AEB44D923CFE}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.Application", "admin\ThingsGateway.Admin.Application\ThingsGateway.Admin.Application.csproj", "{D6685A42-2712-417A-92C5-5EFF90B9FA94}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.ApiController", "admin\ThingsGateway.Admin.ApiController\ThingsGateway.Admin.ApiController.csproj", "{0D17D801-6DAA-4FD1-9A99-F9F07FA6BA88}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "web", "web", "{F0C9A8CB-231B-45E0-B91B-4FEF7EF47197}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		web\Directory.Build.props = web\Directory.Build.props
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Web.Core", "web\ThingsGateway.Web.Core\ThingsGateway.Web.Core.csproj", "{D37EC028-EA46-4510-8261-6E780A906314}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Web.Entry", "web\ThingsGateway.Web.Entry\ThingsGateway.Web.Entry.csproj", "{C5F662EB-991F-438D-BF61-EF87E7371C04}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "解决方案项", "解决方案项", "{97B23D8B-C6C0-4746-A21F-C7B49354B284}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		..\.gitignore = ..\.gitignore
 | 
			
		||||
		..\README.md = ..\README.md
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Components", "Web\ThingsGateway.Components\ThingsGateway.Components.csproj", "{799C49A4-8E23-475A-A82D-080854718BEE}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation", "foundation\ThingsGateway.Foundation\ThingsGateway.Foundation.csproj", "{6961511A-8787-42AF-827D-B630B2AF4791}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.Core", "Web\ThingsGateway.Admin.Core\ThingsGateway.Admin.Core.csproj", "{616CA361-B667-42C8-B4DC-097C7CD39830}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.Application", "Web\ThingsGateway.Admin.Application\ThingsGateway.Admin.Application.csproj", "{16C62A28-BACE-4391-91F8-C2D78D063A1E}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.ApiController", "Web\ThingsGateway.Admin.ApiController\ThingsGateway.Admin.ApiController.csproj", "{8FA03089-322F-44CB-8E4B-F2637388E944}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.Blazor", "Web\ThingsGateway.Admin.Blazor\ThingsGateway.Admin.Blazor.csproj", "{14FF7150-6DB7-455B-AD00-6AB4DE37855B}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Web.Entry", "Web\ThingsGateway.Web.Entry\ThingsGateway.Web.Entry.csproj", "{2861AA39-AAAE-47ED-9ACC-4C165DDF3EF1}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Web.Core", "Web\ThingsGateway.Web.Core\ThingsGateway.Web.Core.csproj", "{9FF2A8A6-48D0-4D8A-9EAD-1905174291CC}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Core", "Web\ThingsGateway.Core\ThingsGateway.Core.csproj", "{51313113-7BB8-494E-9C24-6787BECE39BB}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Admin", "Admin", "{79E7042F-F9E3-4D87-BFA9-4B7DD9736735}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Core", "Core", "{BB9C2A85-7A8A-4CF9-BF44-34DE9848EC15}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{E8D2CBBA-9BCC-44E0-B192-BA0214010203}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		.editorconfig = .editorconfig
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Foundation", "Foundation", "{B6288ADE-A570-4962-8907-991B0FF2D89C}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		Foundation\Directory.Build.props = Foundation\Directory.Build.props
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "foundation", "foundation", "{268A1A81-2685-47E1-9986-5934A58A31A4}"
 | 
			
		||||
EndProject
 | 
			
		||||
Global
 | 
			
		||||
	GlobalSection(SolutionConfigurationPlatforms) = preSolution
 | 
			
		||||
@@ -52,58 +44,56 @@ Global
 | 
			
		||||
		Release|Any CPU = Release|Any CPU
 | 
			
		||||
	EndGlobalSection
 | 
			
		||||
	GlobalSection(ProjectConfigurationPlatforms) = postSolution
 | 
			
		||||
		{2070116F-7A1B-47E2-A2F2-7BEC29ECE891}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{2070116F-7A1B-47E2-A2F2-7BEC29ECE891}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{2070116F-7A1B-47E2-A2F2-7BEC29ECE891}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{2070116F-7A1B-47E2-A2F2-7BEC29ECE891}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{616CA361-B667-42C8-B4DC-097C7CD39830}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{616CA361-B667-42C8-B4DC-097C7CD39830}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{616CA361-B667-42C8-B4DC-097C7CD39830}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{616CA361-B667-42C8-B4DC-097C7CD39830}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{16C62A28-BACE-4391-91F8-C2D78D063A1E}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{16C62A28-BACE-4391-91F8-C2D78D063A1E}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{16C62A28-BACE-4391-91F8-C2D78D063A1E}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{16C62A28-BACE-4391-91F8-C2D78D063A1E}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{8FA03089-322F-44CB-8E4B-F2637388E944}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{8FA03089-322F-44CB-8E4B-F2637388E944}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{8FA03089-322F-44CB-8E4B-F2637388E944}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{8FA03089-322F-44CB-8E4B-F2637388E944}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{14FF7150-6DB7-455B-AD00-6AB4DE37855B}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{14FF7150-6DB7-455B-AD00-6AB4DE37855B}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{14FF7150-6DB7-455B-AD00-6AB4DE37855B}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{14FF7150-6DB7-455B-AD00-6AB4DE37855B}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{2861AA39-AAAE-47ED-9ACC-4C165DDF3EF1}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{2861AA39-AAAE-47ED-9ACC-4C165DDF3EF1}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{2861AA39-AAAE-47ED-9ACC-4C165DDF3EF1}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{2861AA39-AAAE-47ED-9ACC-4C165DDF3EF1}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{9FF2A8A6-48D0-4D8A-9EAD-1905174291CC}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{9FF2A8A6-48D0-4D8A-9EAD-1905174291CC}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{9FF2A8A6-48D0-4D8A-9EAD-1905174291CC}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{9FF2A8A6-48D0-4D8A-9EAD-1905174291CC}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{0A891D8E-23B3-46AD-8D30-565EE5004F93}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{0A891D8E-23B3-46AD-8D30-565EE5004F93}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{0A891D8E-23B3-46AD-8D30-565EE5004F93}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{0A891D8E-23B3-46AD-8D30-565EE5004F93}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{A712EAEE-94F2-4F01-8C1C-2EC802280DD7}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{A712EAEE-94F2-4F01-8C1C-2EC802280DD7}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{A712EAEE-94F2-4F01-8C1C-2EC802280DD7}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{A712EAEE-94F2-4F01-8C1C-2EC802280DD7}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{5DA3D2BD-6768-4479-B52F-49E022EFF310}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{5DA3D2BD-6768-4479-B52F-49E022EFF310}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{5DA3D2BD-6768-4479-B52F-49E022EFF310}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{5DA3D2BD-6768-4479-B52F-49E022EFF310}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{8DD5DF98-7FDE-4B49-8661-AEB44D923CFE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{8DD5DF98-7FDE-4B49-8661-AEB44D923CFE}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{8DD5DF98-7FDE-4B49-8661-AEB44D923CFE}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{8DD5DF98-7FDE-4B49-8661-AEB44D923CFE}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{D6685A42-2712-417A-92C5-5EFF90B9FA94}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{D6685A42-2712-417A-92C5-5EFF90B9FA94}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{D6685A42-2712-417A-92C5-5EFF90B9FA94}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{D6685A42-2712-417A-92C5-5EFF90B9FA94}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{0D17D801-6DAA-4FD1-9A99-F9F07FA6BA88}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{0D17D801-6DAA-4FD1-9A99-F9F07FA6BA88}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{0D17D801-6DAA-4FD1-9A99-F9F07FA6BA88}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{0D17D801-6DAA-4FD1-9A99-F9F07FA6BA88}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{D37EC028-EA46-4510-8261-6E780A906314}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{D37EC028-EA46-4510-8261-6E780A906314}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{D37EC028-EA46-4510-8261-6E780A906314}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{D37EC028-EA46-4510-8261-6E780A906314}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{C5F662EB-991F-438D-BF61-EF87E7371C04}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{C5F662EB-991F-438D-BF61-EF87E7371C04}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{C5F662EB-991F-438D-BF61-EF87E7371C04}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{C5F662EB-991F-438D-BF61-EF87E7371C04}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{6961511A-8787-42AF-827D-B630B2AF4791}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{6961511A-8787-42AF-827D-B630B2AF4791}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{6961511A-8787-42AF-827D-B630B2AF4791}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{6961511A-8787-42AF-827D-B630B2AF4791}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
	EndGlobalSection
 | 
			
		||||
	GlobalSection(SolutionProperties) = preSolution
 | 
			
		||||
		HideSolutionNode = FALSE
 | 
			
		||||
	EndGlobalSection
 | 
			
		||||
	GlobalSection(NestedProjects) = preSolution
 | 
			
		||||
		{2070116F-7A1B-47E2-A2F2-7BEC29ECE891} = {B6288ADE-A570-4962-8907-991B0FF2D89C}
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE} = {BB9C2A85-7A8A-4CF9-BF44-34DE9848EC15}
 | 
			
		||||
		{616CA361-B667-42C8-B4DC-097C7CD39830} = {79E7042F-F9E3-4D87-BFA9-4B7DD9736735}
 | 
			
		||||
		{16C62A28-BACE-4391-91F8-C2D78D063A1E} = {79E7042F-F9E3-4D87-BFA9-4B7DD9736735}
 | 
			
		||||
		{8FA03089-322F-44CB-8E4B-F2637388E944} = {79E7042F-F9E3-4D87-BFA9-4B7DD9736735}
 | 
			
		||||
		{14FF7150-6DB7-455B-AD00-6AB4DE37855B} = {79E7042F-F9E3-4D87-BFA9-4B7DD9736735}
 | 
			
		||||
		{2861AA39-AAAE-47ED-9ACC-4C165DDF3EF1} = {9EB46BB6-D4EA-4B06-95EE-6C971E653030}
 | 
			
		||||
		{9FF2A8A6-48D0-4D8A-9EAD-1905174291CC} = {9EB46BB6-D4EA-4B06-95EE-6C971E653030}
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB} = {BB9C2A85-7A8A-4CF9-BF44-34DE9848EC15}
 | 
			
		||||
		{79E7042F-F9E3-4D87-BFA9-4B7DD9736735} = {9EB46BB6-D4EA-4B06-95EE-6C971E653030}
 | 
			
		||||
		{BB9C2A85-7A8A-4CF9-BF44-34DE9848EC15} = {9EB46BB6-D4EA-4B06-95EE-6C971E653030}
 | 
			
		||||
		{0A891D8E-23B3-46AD-8D30-565EE5004F93} = {4E66C22C-0636-4949-BF6A-9E3BBE1550BA}
 | 
			
		||||
		{A712EAEE-94F2-4F01-8C1C-2EC802280DD7} = {4E66C22C-0636-4949-BF6A-9E3BBE1550BA}
 | 
			
		||||
		{5DA3D2BD-6768-4479-B52F-49E022EFF310} = {4E66C22C-0636-4949-BF6A-9E3BBE1550BA}
 | 
			
		||||
		{8DD5DF98-7FDE-4B49-8661-AEB44D923CFE} = {4E66C22C-0636-4949-BF6A-9E3BBE1550BA}
 | 
			
		||||
		{D6685A42-2712-417A-92C5-5EFF90B9FA94} = {4E66C22C-0636-4949-BF6A-9E3BBE1550BA}
 | 
			
		||||
		{0D17D801-6DAA-4FD1-9A99-F9F07FA6BA88} = {4E66C22C-0636-4949-BF6A-9E3BBE1550BA}
 | 
			
		||||
		{D37EC028-EA46-4510-8261-6E780A906314} = {F0C9A8CB-231B-45E0-B91B-4FEF7EF47197}
 | 
			
		||||
		{C5F662EB-991F-438D-BF61-EF87E7371C04} = {F0C9A8CB-231B-45E0-B91B-4FEF7EF47197}
 | 
			
		||||
		{6961511A-8787-42AF-827D-B630B2AF4791} = {268A1A81-2685-47E1-9986-5934A58A31A4}
 | 
			
		||||
	EndGlobalSection
 | 
			
		||||
	GlobalSection(ExtensibilityGlobals) = postSolution
 | 
			
		||||
		SolutionGuid = {C49B2D3E-6818-4E28-91B7-6E4E7E264BBB}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,49 +0,0 @@
 | 
			
		||||
 | 
			
		||||
Microsoft Visual Studio Solution File, Format Version 12.00
 | 
			
		||||
# Visual Studio Version 17
 | 
			
		||||
VisualStudioVersion = 17.6.33927.249
 | 
			
		||||
MinimumVisualStudioVersion = 10.0.40219.1
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation", "Foundation\ThingsGateway.Foundation\ThingsGateway.Foundation.csproj", "{2070116F-7A1B-47E2-A2F2-7BEC29ECE891}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Components", "Web\ThingsGateway.Components\ThingsGateway.Components.csproj", "{799C49A4-8E23-475A-A82D-080854718BEE}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Core", "Web\ThingsGateway.Core\ThingsGateway.Core.csproj", "{51313113-7BB8-494E-9C24-6787BECE39BB}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.UpgradeManger", "UpgradeManger\ThingsGateway.UpgradeManger\ThingsGateway.UpgradeManger.csproj", "{84362F1D-E788-4646-B555-5A3629B55EFC}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Upgrade", "UpgradeManger\ThingsGateway.Upgrade\ThingsGateway.Upgrade.csproj", "{681F774F-7B0B-450A-917C-1385E1847CA6}"
 | 
			
		||||
EndProject
 | 
			
		||||
Global
 | 
			
		||||
	GlobalSection(SolutionConfigurationPlatforms) = preSolution
 | 
			
		||||
		Debug|Any CPU = Debug|Any CPU
 | 
			
		||||
		Release|Any CPU = Release|Any CPU
 | 
			
		||||
	EndGlobalSection
 | 
			
		||||
	GlobalSection(ProjectConfigurationPlatforms) = postSolution
 | 
			
		||||
		{2070116F-7A1B-47E2-A2F2-7BEC29ECE891}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{2070116F-7A1B-47E2-A2F2-7BEC29ECE891}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{2070116F-7A1B-47E2-A2F2-7BEC29ECE891}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{2070116F-7A1B-47E2-A2F2-7BEC29ECE891}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{84362F1D-E788-4646-B555-5A3629B55EFC}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{84362F1D-E788-4646-B555-5A3629B55EFC}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{84362F1D-E788-4646-B555-5A3629B55EFC}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{84362F1D-E788-4646-B555-5A3629B55EFC}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{681F774F-7B0B-450A-917C-1385E1847CA6}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{681F774F-7B0B-450A-917C-1385E1847CA6}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{681F774F-7B0B-450A-917C-1385E1847CA6}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{681F774F-7B0B-450A-917C-1385E1847CA6}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
	EndGlobalSection
 | 
			
		||||
	GlobalSection(SolutionProperties) = preSolution
 | 
			
		||||
		HideSolutionNode = FALSE
 | 
			
		||||
	EndGlobalSection
 | 
			
		||||
	GlobalSection(ExtensibilityGlobals) = postSolution
 | 
			
		||||
		SolutionGuid = {C49B2D3E-6818-4E28-91B7-6E4E7E264BBB}
 | 
			
		||||
	EndGlobalSection
 | 
			
		||||
EndGlobal
 | 
			
		||||
@@ -3,107 +3,108 @@ Microsoft Visual Studio Solution File, Format Version 12.00
 | 
			
		||||
# Visual Studio Version 17
 | 
			
		||||
VisualStudioVersion = 17.6.33927.249
 | 
			
		||||
MinimumVisualStudioVersion = 10.0.40219.1
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation", "Foundation\ThingsGateway.Foundation\ThingsGateway.Foundation.csproj", "{2070116F-7A1B-47E2-A2F2-7BEC29ECE891}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation", "foundation\ThingsGateway.foundation\ThingsGateway.Foundation.csproj", "{2070116F-7A1B-47E2-A2F2-7BEC29ECE891}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Foundation", "Foundation", "{0874CBC5-C583-4FAD-BA93-94571D446898}"
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "foundation", "foundation", "{0874CBC5-C583-4FAD-BA93-94571D446898}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		Foundation\Directory.Build.props = Foundation\Directory.Build.props
 | 
			
		||||
		foundation\Directory.Build.props = foundation\Directory.Build.props
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Adapter.DLT645", "Foundation\ThingsGateway.Foundation.Adapter.DLT645\ThingsGateway.Foundation.Adapter.DLT645.csproj", "{92963877-0185-45A5-A0EB-CEC0D55FF9B0}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Adapter.DLT645", "foundation\ThingsGateway.Foundation.Adapter.DLT645\ThingsGateway.Foundation.Adapter.DLT645.csproj", "{92963877-0185-45A5-A0EB-CEC0D55FF9B0}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Adapter.Modbus", "Foundation\ThingsGateway.Foundation.Adapter.Modbus\ThingsGateway.Foundation.Adapter.Modbus.csproj", "{24510CCE-0B60-4A03-9719-CF367C3528E9}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Adapter.Modbus", "foundation\ThingsGateway.Foundation.Adapter.Modbus\ThingsGateway.Foundation.Adapter.Modbus.csproj", "{24510CCE-0B60-4A03-9719-CF367C3528E9}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Adapter.OPCDA", "Foundation\ThingsGateway.Foundation.Adapter.OPCDA\ThingsGateway.Foundation.Adapter.OPCDA.csproj", "{566783A4-222B-46F5-AA12-0753997B3254}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Adapter.OPCDA", "foundation\ThingsGateway.Foundation.Adapter.OPCDA\ThingsGateway.Foundation.Adapter.OPCDA.csproj", "{566783A4-222B-46F5-AA12-0753997B3254}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Adapter.OPCUA", "Foundation\ThingsGateway.Foundation.Adapter.OPCUA\ThingsGateway.Foundation.Adapter.OPCUA.csproj", "{BEFBC44A-E140-4E3E-AFBE-2DD8B98EB9BF}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Adapter.OPCUA", "foundation\ThingsGateway.Foundation.Adapter.OPCUA\ThingsGateway.Foundation.Adapter.OPCUA.csproj", "{BEFBC44A-E140-4E3E-AFBE-2DD8B98EB9BF}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Adapter.Siemens", "Foundation\ThingsGateway.Foundation.Adapter.Siemens\ThingsGateway.Foundation.Adapter.Siemens.csproj", "{9695B353-D773-40DD-B65E-7B10EB0C16EC}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Adapter.Siemens", "foundation\ThingsGateway.Foundation.Adapter.Siemens\ThingsGateway.Foundation.Adapter.Siemens.csproj", "{9695B353-D773-40DD-B65E-7B10EB0C16EC}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Demo", "Demo", "{95008B83-0324-4A7C-80DE-2BBDDD1A9099}"
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "demo", "demo", "{358AE1E1-4DD6-4D1F-8935-1ACBB0111A01}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		Demo\Directory.Build.props = Demo\Directory.Build.props
 | 
			
		||||
		demo\Directory.Build.props = demo\Directory.Build.props
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Web", "Web", "{9EB46BB6-D4EA-4B06-95EE-6C971E653030}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Demo.Photino", "demo\ThingsGateway.Foundation.Demo.Photino\ThingsGateway.Foundation.Demo.Photino.csproj", "{2192217C-CF77-422E-9E63-DF4003ABF01A}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Demo.Rcl", "demo\ThingsGateway.Foundation.Demo.Rcl\ThingsGateway.Foundation.Demo.Rcl.csproj", "{DE376FAD-1E99-4CEE-96B2-08B1D1F3D1DB}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Demo.Winform", "demo\ThingsGateway.Foundation.Demo.Winform\ThingsGateway.Foundation.Demo.Winform.csproj", "{ACAA3F0D-CE2B-49F4-883A-ED23ADD9E325}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "gateway", "gateway", "{DFB97103-FC3A-4DC3-A327-DA8C65BDA607}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		Web\Directory.Build.props = Web\Directory.Build.props
 | 
			
		||||
		gateway\Directory.Build.props = gateway\Directory.Build.props
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "script", "script", "{4A64518E-C072-4607-BBF7-7D392CEC9D58}"
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "admin", "admin", "{4E66C22C-0636-4949-BF6A-9E3BBE1550BA}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		admin\Directory.Build.props = admin\Directory.Build.props
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Components", "admin\ThingsGateway.Components\ThingsGateway.Components.csproj", "{0A891D8E-23B3-46AD-8D30-565EE5004F93}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Core", "admin\ThingsGateway.Core\ThingsGateway.Core.csproj", "{A712EAEE-94F2-4F01-8C1C-2EC802280DD7}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.Core", "admin\ThingsGateway.Admin.Core\ThingsGateway.Admin.Core.csproj", "{5DA3D2BD-6768-4479-B52F-49E022EFF310}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.Blazor", "admin\ThingsGateway.Admin.Blazor\ThingsGateway.Admin.Blazor.csproj", "{8DD5DF98-7FDE-4B49-8661-AEB44D923CFE}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.Application", "admin\ThingsGateway.Admin.Application\ThingsGateway.Admin.Application.csproj", "{D6685A42-2712-417A-92C5-5EFF90B9FA94}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.ApiController", "admin\ThingsGateway.Admin.ApiController\ThingsGateway.Admin.ApiController.csproj", "{0D17D801-6DAA-4FD1-9A99-F9F07FA6BA88}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Gateway.Core", "gateway\ThingsGateway.Gateway.Core\ThingsGateway.Gateway.Core.csproj", "{1A4D0B95-9FC5-4687-94E2-B6F86B5427F2}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Gateway.Blazor", "gateway\ThingsGateway.Gateway.Blazor\ThingsGateway.Gateway.Blazor.csproj", "{69EA3D89-CB40-425A-8D70-5E4A33337BE5}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Gateway.Application", "gateway\ThingsGateway.Gateway.Application\ThingsGateway.Gateway.Application.csproj", "{90AB5C24-1AA3-4F58-9987-B307B92B5193}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Gateway.ApiController", "gateway\ThingsGateway.Gateway.ApiController\ThingsGateway.Gateway.ApiController.csproj", "{4C7A5A90-8292-413B-8848-419D9DCDE66F}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "web", "web", "{F0C9A8CB-231B-45E0-B91B-4FEF7EF47197}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		web\Directory.Build.props = web\Directory.Build.props
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Web.Core", "web\ThingsGateway.Web.Core\ThingsGateway.Web.Core.csproj", "{D37EC028-EA46-4510-8261-6E780A906314}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Web.Entry", "web\ThingsGateway.Web.Entry\ThingsGateway.Web.Entry.csproj", "{C5F662EB-991F-438D-BF61-EF87E7371C04}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "plugin", "plugin", "{E65490B8-D2E2-4693-B39C-15703B1EBFBB}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		plugin\Directory.Build.props = plugin\Directory.Build.props
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.Modbus", "plugin\ThingsGateway.Plugin.Modbus\ThingsGateway.Plugin.Modbus.csproj", "{ECF3A7A3-2363-4A38-BC78-8FF9A8564603}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.Mqtt", "plugin\ThingsGateway.Plugin.Mqtt\ThingsGateway.Plugin.Mqtt.csproj", "{53C870AE-0249-4485-AE63-F8A16EA4BCB4}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "解决方案项", "解决方案项", "{97B23D8B-C6C0-4746-A21F-C7B49354B284}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		..\.gitignore = ..\.gitignore
 | 
			
		||||
		Directory.Build.props = Directory.Build.props
 | 
			
		||||
		..\README.md = ..\README.md
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Components", "Web\ThingsGateway.Components\ThingsGateway.Components.csproj", "{799C49A4-8E23-475A-A82D-080854718BEE}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.Kafka", "plugin\ThingsGateway.Plugin.Kafka\ThingsGateway.Plugin.Kafka.csproj", "{E4B8B8E6-FAE7-43BA-9A51-33A3CD9FB825}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.Core", "Web\ThingsGateway.Admin.Core\ThingsGateway.Admin.Core.csproj", "{616CA361-B667-42C8-B4DC-097C7CD39830}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.SQLDB", "plugin\ThingsGateway.Plugin.SQLDB\ThingsGateway.Plugin.SQLDB.csproj", "{C102AB3A-377E-4753-AFA7-C13250D7DF00}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.Application", "Web\ThingsGateway.Admin.Application\ThingsGateway.Admin.Application.csproj", "{16C62A28-BACE-4391-91F8-C2D78D063A1E}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.QuestDB", "plugin\ThingsGateway.Plugin.QuestDB\ThingsGateway.Plugin.QuestDB.csproj", "{0DE1D254-0BD2-4D98-B939-5440BE5EB552}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.ApiController", "Web\ThingsGateway.Admin.ApiController\ThingsGateway.Admin.ApiController.csproj", "{8FA03089-322F-44CB-8E4B-F2637388E944}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.TDengineDB", "plugin\ThingsGateway.Plugin.TDengineDB\ThingsGateway.Plugin.TDengineDB.csproj", "{C1911F84-6FDB-4DEC-8938-69B2E3901CC4}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Admin.Blazor", "Web\ThingsGateway.Admin.Blazor\ThingsGateway.Admin.Blazor.csproj", "{14FF7150-6DB7-455B-AD00-6AB4DE37855B}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.RabbitMQ", "plugin\ThingsGateway.Plugin.RabbitMQ\ThingsGateway.Plugin.RabbitMQ.csproj", "{A8E68E17-4EBF-4E4C-8272-B489329A68BF}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Web.Entry", "Web\ThingsGateway.Web.Entry\ThingsGateway.Web.Entry.csproj", "{2861AA39-AAAE-47ED-9ACC-4C165DDF3EF1}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.Siemens", "plugin\ThingsGateway.Plugin.Siemens\ThingsGateway.Plugin.Siemens.csproj", "{5FAFD46E-6B98-4C75-B1EB-085AA18F15FD}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Web.Core", "Web\ThingsGateway.Web.Core\ThingsGateway.Web.Core.csproj", "{9FF2A8A6-48D0-4D8A-9EAD-1905174291CC}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.DLT645", "plugin\ThingsGateway.Plugin.DLT645\ThingsGateway.Plugin.DLT645.csproj", "{AC4295F2-AB2F-4137-99EF-80FA5C83896B}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Core", "Web\ThingsGateway.Core\ThingsGateway.Core.csproj", "{51313113-7BB8-494E-9C24-6787BECE39BB}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.OPCDA", "plugin\ThingsGateway.Plugin.OPCDA\ThingsGateway.Plugin.OPCDA.csproj", "{9DA9AED3-9572-4378-A2A6-4D792D67ADDC}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Admin", "Admin", "{79E7042F-F9E3-4D87-BFA9-4B7DD9736735}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.OPCUA", "plugin\ThingsGateway.Plugin.OPCUA\ThingsGateway.Plugin.OPCUA.csproj", "{FD4969DB-3CCE-4CCF-BAE4-1BE8A3F40812}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Gateway", "Gateway", "{000C3C62-345E-451C-8CEE-6F2C6A087116}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Gateway.ApiController", "Web\ThingsGateway.Gateway.ApiController\ThingsGateway.Gateway.ApiController.csproj", "{5D7BE567-2345-46C8-9F54-DDC1DA96D198}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Gateway.Application", "Web\ThingsGateway.Gateway.Application\ThingsGateway.Gateway.Application.csproj", "{5CF1B3EC-84E2-484A-8DFC-2ECD2EE18E2F}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Gateway.Blazor", "Web\ThingsGateway.Gateway.Blazor\ThingsGateway.Gateway.Blazor.csproj", "{CD0F211A-F65B-4026-9750-68AC3C70D012}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Gateway.Core", "Web\ThingsGateway.Gateway.Core\ThingsGateway.Gateway.Core.csproj", "{5CD79F91-7182-4A9D-9BEF-4DF410C782D2}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Core", "Core", "{BB9C2A85-7A8A-4CF9-BF44-34DE9848EC15}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Plugin", "Plugin", "{CC8D0880-B73E-4DFC-9052-86504728708E}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		Plugin\Directory.Build.props = Plugin\Directory.Build.props
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.Modbus", "Plugin\ThingsGateway.Plugin.Modbus\ThingsGateway.Plugin.Modbus.csproj", "{2057E5BE-FACA-4D44-A2BA-E1F864A8DAFF}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.DLT645", "Plugin\ThingsGateway.Plugin.DLT645\ThingsGateway.Plugin.DLT645.csproj", "{A723D4D7-B796-4D97-BA68-95E5696C9559}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.Kafka", "Plugin\ThingsGateway.Plugin.Kafka\ThingsGateway.Plugin.Kafka.csproj", "{D9944D52-81B4-4DBC-8C3B-2A334CCBD4F6}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.Mqtt", "Plugin\ThingsGateway.Plugin.Mqtt\ThingsGateway.Plugin.Mqtt.csproj", "{C5C89B2F-D9C9-4A67-A9BE-4E5F0B42162F}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.OPCDA", "Plugin\ThingsGateway.Plugin.OPCDA\ThingsGateway.Plugin.OPCDA.csproj", "{7562C85D-267D-4718-8857-422B625C5BD0}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.OPCUA", "Plugin\ThingsGateway.Plugin.OPCUA\ThingsGateway.Plugin.OPCUA.csproj", "{A0E1EAC7-10DD-4520-973F-CC385BBFBD84}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.RabbitMQ", "Plugin\ThingsGateway.Plugin.RabbitMQ\ThingsGateway.Plugin.RabbitMQ.csproj", "{503529C4-842A-47F0-928B-C8DC5B0A367D}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.Siemens", "Plugin\ThingsGateway.Plugin.Siemens\ThingsGateway.Plugin.Siemens.csproj", "{8A9F6586-3320-4F03-B3DA-09BF39AA90A0}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{E8D2CBBA-9BCC-44E0-B192-BA0214010203}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		.editorconfig = .editorconfig
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "UpgradeManger", "UpgradeManger", "{237C7BC5-7B07-40B5-AF42-CE2F8E0893C3}"
 | 
			
		||||
	ProjectSection(SolutionItems) = preProject
 | 
			
		||||
		UpgradeManger\Directory.Build.props = UpgradeManger\Directory.Build.props
 | 
			
		||||
	EndProjectSection
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.UpgradeManger", "UpgradeManger\ThingsGateway.UpgradeManger\ThingsGateway.UpgradeManger.csproj", "{84362F1D-E788-4646-B555-5A3629B55EFC}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Upgrade", "UpgradeManger\ThingsGateway.Upgrade\ThingsGateway.Upgrade.csproj", "{681F774F-7B0B-450A-917C-1385E1847CA6}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Demo.Rcl", "Demo\ThingsGateway.Foundation.Demo.Rcl\ThingsGateway.Foundation.Demo.Rcl.csproj", "{637A662B-7B70-4CE8-8F5F-0A095B9D77EC}"
 | 
			
		||||
EndProject
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Foundation.Demo.Photino", "Demo\ThingsGateway.Foundation.Demo.Photino\ThingsGateway.Foundation.Demo.Photino.csproj", "{C5519C51-0A0C-4317-A43D-FFBB6B344ACB}"
 | 
			
		||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ThingsGateway.Plugin.SQLHisAlarm", "plugin\ThingsGateway.Plugin.SQLHisAlarm\ThingsGateway.Plugin.SQLHisAlarm.csproj", "{0947E6C0-6371-4483-B0ED-191FB5073151}"
 | 
			
		||||
EndProject
 | 
			
		||||
Global
 | 
			
		||||
	GlobalSection(SolutionConfigurationPlatforms) = preSolution
 | 
			
		||||
@@ -135,102 +136,114 @@ Global
 | 
			
		||||
		{9695B353-D773-40DD-B65E-7B10EB0C16EC}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{9695B353-D773-40DD-B65E-7B10EB0C16EC}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{9695B353-D773-40DD-B65E-7B10EB0C16EC}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{616CA361-B667-42C8-B4DC-097C7CD39830}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{616CA361-B667-42C8-B4DC-097C7CD39830}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{616CA361-B667-42C8-B4DC-097C7CD39830}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{616CA361-B667-42C8-B4DC-097C7CD39830}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{16C62A28-BACE-4391-91F8-C2D78D063A1E}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{16C62A28-BACE-4391-91F8-C2D78D063A1E}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{16C62A28-BACE-4391-91F8-C2D78D063A1E}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{16C62A28-BACE-4391-91F8-C2D78D063A1E}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{8FA03089-322F-44CB-8E4B-F2637388E944}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{8FA03089-322F-44CB-8E4B-F2637388E944}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{8FA03089-322F-44CB-8E4B-F2637388E944}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{8FA03089-322F-44CB-8E4B-F2637388E944}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{14FF7150-6DB7-455B-AD00-6AB4DE37855B}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{14FF7150-6DB7-455B-AD00-6AB4DE37855B}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{14FF7150-6DB7-455B-AD00-6AB4DE37855B}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{14FF7150-6DB7-455B-AD00-6AB4DE37855B}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{2861AA39-AAAE-47ED-9ACC-4C165DDF3EF1}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{2861AA39-AAAE-47ED-9ACC-4C165DDF3EF1}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{2861AA39-AAAE-47ED-9ACC-4C165DDF3EF1}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{2861AA39-AAAE-47ED-9ACC-4C165DDF3EF1}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{9FF2A8A6-48D0-4D8A-9EAD-1905174291CC}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{9FF2A8A6-48D0-4D8A-9EAD-1905174291CC}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{9FF2A8A6-48D0-4D8A-9EAD-1905174291CC}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{9FF2A8A6-48D0-4D8A-9EAD-1905174291CC}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{5D7BE567-2345-46C8-9F54-DDC1DA96D198}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{5D7BE567-2345-46C8-9F54-DDC1DA96D198}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{5D7BE567-2345-46C8-9F54-DDC1DA96D198}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{5D7BE567-2345-46C8-9F54-DDC1DA96D198}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{5CF1B3EC-84E2-484A-8DFC-2ECD2EE18E2F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{5CF1B3EC-84E2-484A-8DFC-2ECD2EE18E2F}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{5CF1B3EC-84E2-484A-8DFC-2ECD2EE18E2F}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{5CF1B3EC-84E2-484A-8DFC-2ECD2EE18E2F}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{CD0F211A-F65B-4026-9750-68AC3C70D012}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{CD0F211A-F65B-4026-9750-68AC3C70D012}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{CD0F211A-F65B-4026-9750-68AC3C70D012}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{CD0F211A-F65B-4026-9750-68AC3C70D012}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{5CD79F91-7182-4A9D-9BEF-4DF410C782D2}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{5CD79F91-7182-4A9D-9BEF-4DF410C782D2}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{5CD79F91-7182-4A9D-9BEF-4DF410C782D2}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{5CD79F91-7182-4A9D-9BEF-4DF410C782D2}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{2057E5BE-FACA-4D44-A2BA-E1F864A8DAFF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{2057E5BE-FACA-4D44-A2BA-E1F864A8DAFF}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{2057E5BE-FACA-4D44-A2BA-E1F864A8DAFF}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{2057E5BE-FACA-4D44-A2BA-E1F864A8DAFF}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{A723D4D7-B796-4D97-BA68-95E5696C9559}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{A723D4D7-B796-4D97-BA68-95E5696C9559}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{A723D4D7-B796-4D97-BA68-95E5696C9559}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{A723D4D7-B796-4D97-BA68-95E5696C9559}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{D9944D52-81B4-4DBC-8C3B-2A334CCBD4F6}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{D9944D52-81B4-4DBC-8C3B-2A334CCBD4F6}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{D9944D52-81B4-4DBC-8C3B-2A334CCBD4F6}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{D9944D52-81B4-4DBC-8C3B-2A334CCBD4F6}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{C5C89B2F-D9C9-4A67-A9BE-4E5F0B42162F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{C5C89B2F-D9C9-4A67-A9BE-4E5F0B42162F}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{C5C89B2F-D9C9-4A67-A9BE-4E5F0B42162F}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{C5C89B2F-D9C9-4A67-A9BE-4E5F0B42162F}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{7562C85D-267D-4718-8857-422B625C5BD0}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{7562C85D-267D-4718-8857-422B625C5BD0}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{7562C85D-267D-4718-8857-422B625C5BD0}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{7562C85D-267D-4718-8857-422B625C5BD0}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{A0E1EAC7-10DD-4520-973F-CC385BBFBD84}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{A0E1EAC7-10DD-4520-973F-CC385BBFBD84}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{A0E1EAC7-10DD-4520-973F-CC385BBFBD84}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{A0E1EAC7-10DD-4520-973F-CC385BBFBD84}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{503529C4-842A-47F0-928B-C8DC5B0A367D}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{503529C4-842A-47F0-928B-C8DC5B0A367D}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{503529C4-842A-47F0-928B-C8DC5B0A367D}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{503529C4-842A-47F0-928B-C8DC5B0A367D}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{8A9F6586-3320-4F03-B3DA-09BF39AA90A0}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{8A9F6586-3320-4F03-B3DA-09BF39AA90A0}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{8A9F6586-3320-4F03-B3DA-09BF39AA90A0}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{8A9F6586-3320-4F03-B3DA-09BF39AA90A0}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{84362F1D-E788-4646-B555-5A3629B55EFC}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{84362F1D-E788-4646-B555-5A3629B55EFC}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{84362F1D-E788-4646-B555-5A3629B55EFC}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{84362F1D-E788-4646-B555-5A3629B55EFC}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{681F774F-7B0B-450A-917C-1385E1847CA6}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{681F774F-7B0B-450A-917C-1385E1847CA6}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{681F774F-7B0B-450A-917C-1385E1847CA6}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{681F774F-7B0B-450A-917C-1385E1847CA6}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{637A662B-7B70-4CE8-8F5F-0A095B9D77EC}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{637A662B-7B70-4CE8-8F5F-0A095B9D77EC}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{637A662B-7B70-4CE8-8F5F-0A095B9D77EC}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{637A662B-7B70-4CE8-8F5F-0A095B9D77EC}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{C5519C51-0A0C-4317-A43D-FFBB6B344ACB}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{C5519C51-0A0C-4317-A43D-FFBB6B344ACB}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{C5519C51-0A0C-4317-A43D-FFBB6B344ACB}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{C5519C51-0A0C-4317-A43D-FFBB6B344ACB}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{2192217C-CF77-422E-9E63-DF4003ABF01A}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{2192217C-CF77-422E-9E63-DF4003ABF01A}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{2192217C-CF77-422E-9E63-DF4003ABF01A}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{2192217C-CF77-422E-9E63-DF4003ABF01A}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{DE376FAD-1E99-4CEE-96B2-08B1D1F3D1DB}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{DE376FAD-1E99-4CEE-96B2-08B1D1F3D1DB}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{DE376FAD-1E99-4CEE-96B2-08B1D1F3D1DB}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{DE376FAD-1E99-4CEE-96B2-08B1D1F3D1DB}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{ACAA3F0D-CE2B-49F4-883A-ED23ADD9E325}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{ACAA3F0D-CE2B-49F4-883A-ED23ADD9E325}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{ACAA3F0D-CE2B-49F4-883A-ED23ADD9E325}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{ACAA3F0D-CE2B-49F4-883A-ED23ADD9E325}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{0A891D8E-23B3-46AD-8D30-565EE5004F93}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{0A891D8E-23B3-46AD-8D30-565EE5004F93}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{0A891D8E-23B3-46AD-8D30-565EE5004F93}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{0A891D8E-23B3-46AD-8D30-565EE5004F93}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{A712EAEE-94F2-4F01-8C1C-2EC802280DD7}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{A712EAEE-94F2-4F01-8C1C-2EC802280DD7}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{A712EAEE-94F2-4F01-8C1C-2EC802280DD7}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{A712EAEE-94F2-4F01-8C1C-2EC802280DD7}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{5DA3D2BD-6768-4479-B52F-49E022EFF310}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{5DA3D2BD-6768-4479-B52F-49E022EFF310}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{5DA3D2BD-6768-4479-B52F-49E022EFF310}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{5DA3D2BD-6768-4479-B52F-49E022EFF310}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{8DD5DF98-7FDE-4B49-8661-AEB44D923CFE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{8DD5DF98-7FDE-4B49-8661-AEB44D923CFE}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{8DD5DF98-7FDE-4B49-8661-AEB44D923CFE}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{8DD5DF98-7FDE-4B49-8661-AEB44D923CFE}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{D6685A42-2712-417A-92C5-5EFF90B9FA94}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{D6685A42-2712-417A-92C5-5EFF90B9FA94}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{D6685A42-2712-417A-92C5-5EFF90B9FA94}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{D6685A42-2712-417A-92C5-5EFF90B9FA94}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{0D17D801-6DAA-4FD1-9A99-F9F07FA6BA88}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{0D17D801-6DAA-4FD1-9A99-F9F07FA6BA88}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{0D17D801-6DAA-4FD1-9A99-F9F07FA6BA88}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{0D17D801-6DAA-4FD1-9A99-F9F07FA6BA88}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{1A4D0B95-9FC5-4687-94E2-B6F86B5427F2}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{1A4D0B95-9FC5-4687-94E2-B6F86B5427F2}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{1A4D0B95-9FC5-4687-94E2-B6F86B5427F2}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{1A4D0B95-9FC5-4687-94E2-B6F86B5427F2}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{69EA3D89-CB40-425A-8D70-5E4A33337BE5}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{69EA3D89-CB40-425A-8D70-5E4A33337BE5}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{69EA3D89-CB40-425A-8D70-5E4A33337BE5}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{69EA3D89-CB40-425A-8D70-5E4A33337BE5}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{90AB5C24-1AA3-4F58-9987-B307B92B5193}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{90AB5C24-1AA3-4F58-9987-B307B92B5193}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{90AB5C24-1AA3-4F58-9987-B307B92B5193}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{90AB5C24-1AA3-4F58-9987-B307B92B5193}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{4C7A5A90-8292-413B-8848-419D9DCDE66F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{4C7A5A90-8292-413B-8848-419D9DCDE66F}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{4C7A5A90-8292-413B-8848-419D9DCDE66F}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{4C7A5A90-8292-413B-8848-419D9DCDE66F}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{D37EC028-EA46-4510-8261-6E780A906314}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{D37EC028-EA46-4510-8261-6E780A906314}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{D37EC028-EA46-4510-8261-6E780A906314}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{D37EC028-EA46-4510-8261-6E780A906314}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{C5F662EB-991F-438D-BF61-EF87E7371C04}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{C5F662EB-991F-438D-BF61-EF87E7371C04}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{C5F662EB-991F-438D-BF61-EF87E7371C04}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{C5F662EB-991F-438D-BF61-EF87E7371C04}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{ECF3A7A3-2363-4A38-BC78-8FF9A8564603}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{ECF3A7A3-2363-4A38-BC78-8FF9A8564603}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{ECF3A7A3-2363-4A38-BC78-8FF9A8564603}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{ECF3A7A3-2363-4A38-BC78-8FF9A8564603}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{53C870AE-0249-4485-AE63-F8A16EA4BCB4}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{53C870AE-0249-4485-AE63-F8A16EA4BCB4}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{53C870AE-0249-4485-AE63-F8A16EA4BCB4}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{53C870AE-0249-4485-AE63-F8A16EA4BCB4}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{E4B8B8E6-FAE7-43BA-9A51-33A3CD9FB825}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{E4B8B8E6-FAE7-43BA-9A51-33A3CD9FB825}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{E4B8B8E6-FAE7-43BA-9A51-33A3CD9FB825}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{E4B8B8E6-FAE7-43BA-9A51-33A3CD9FB825}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{C102AB3A-377E-4753-AFA7-C13250D7DF00}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{C102AB3A-377E-4753-AFA7-C13250D7DF00}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{C102AB3A-377E-4753-AFA7-C13250D7DF00}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{C102AB3A-377E-4753-AFA7-C13250D7DF00}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{0DE1D254-0BD2-4D98-B939-5440BE5EB552}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{0DE1D254-0BD2-4D98-B939-5440BE5EB552}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{0DE1D254-0BD2-4D98-B939-5440BE5EB552}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{0DE1D254-0BD2-4D98-B939-5440BE5EB552}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{C1911F84-6FDB-4DEC-8938-69B2E3901CC4}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{C1911F84-6FDB-4DEC-8938-69B2E3901CC4}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{C1911F84-6FDB-4DEC-8938-69B2E3901CC4}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{C1911F84-6FDB-4DEC-8938-69B2E3901CC4}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{A8E68E17-4EBF-4E4C-8272-B489329A68BF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{A8E68E17-4EBF-4E4C-8272-B489329A68BF}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{A8E68E17-4EBF-4E4C-8272-B489329A68BF}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{A8E68E17-4EBF-4E4C-8272-B489329A68BF}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{5FAFD46E-6B98-4C75-B1EB-085AA18F15FD}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{5FAFD46E-6B98-4C75-B1EB-085AA18F15FD}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{5FAFD46E-6B98-4C75-B1EB-085AA18F15FD}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{5FAFD46E-6B98-4C75-B1EB-085AA18F15FD}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{AC4295F2-AB2F-4137-99EF-80FA5C83896B}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{AC4295F2-AB2F-4137-99EF-80FA5C83896B}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{AC4295F2-AB2F-4137-99EF-80FA5C83896B}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{AC4295F2-AB2F-4137-99EF-80FA5C83896B}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{9DA9AED3-9572-4378-A2A6-4D792D67ADDC}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{9DA9AED3-9572-4378-A2A6-4D792D67ADDC}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{9DA9AED3-9572-4378-A2A6-4D792D67ADDC}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{9DA9AED3-9572-4378-A2A6-4D792D67ADDC}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{FD4969DB-3CCE-4CCF-BAE4-1BE8A3F40812}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{FD4969DB-3CCE-4CCF-BAE4-1BE8A3F40812}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{FD4969DB-3CCE-4CCF-BAE4-1BE8A3F40812}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{FD4969DB-3CCE-4CCF-BAE4-1BE8A3F40812}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
		{0947E6C0-6371-4483-B0ED-191FB5073151}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
 | 
			
		||||
		{0947E6C0-6371-4483-B0ED-191FB5073151}.Debug|Any CPU.Build.0 = Debug|Any CPU
 | 
			
		||||
		{0947E6C0-6371-4483-B0ED-191FB5073151}.Release|Any CPU.ActiveCfg = Release|Any CPU
 | 
			
		||||
		{0947E6C0-6371-4483-B0ED-191FB5073151}.Release|Any CPU.Build.0 = Release|Any CPU
 | 
			
		||||
	EndGlobalSection
 | 
			
		||||
	GlobalSection(SolutionProperties) = preSolution
 | 
			
		||||
		HideSolutionNode = FALSE
 | 
			
		||||
@@ -242,33 +255,33 @@ Global
 | 
			
		||||
		{566783A4-222B-46F5-AA12-0753997B3254} = {0874CBC5-C583-4FAD-BA93-94571D446898}
 | 
			
		||||
		{BEFBC44A-E140-4E3E-AFBE-2DD8B98EB9BF} = {0874CBC5-C583-4FAD-BA93-94571D446898}
 | 
			
		||||
		{9695B353-D773-40DD-B65E-7B10EB0C16EC} = {0874CBC5-C583-4FAD-BA93-94571D446898}
 | 
			
		||||
		{799C49A4-8E23-475A-A82D-080854718BEE} = {BB9C2A85-7A8A-4CF9-BF44-34DE9848EC15}
 | 
			
		||||
		{616CA361-B667-42C8-B4DC-097C7CD39830} = {79E7042F-F9E3-4D87-BFA9-4B7DD9736735}
 | 
			
		||||
		{16C62A28-BACE-4391-91F8-C2D78D063A1E} = {79E7042F-F9E3-4D87-BFA9-4B7DD9736735}
 | 
			
		||||
		{8FA03089-322F-44CB-8E4B-F2637388E944} = {79E7042F-F9E3-4D87-BFA9-4B7DD9736735}
 | 
			
		||||
		{14FF7150-6DB7-455B-AD00-6AB4DE37855B} = {79E7042F-F9E3-4D87-BFA9-4B7DD9736735}
 | 
			
		||||
		{2861AA39-AAAE-47ED-9ACC-4C165DDF3EF1} = {9EB46BB6-D4EA-4B06-95EE-6C971E653030}
 | 
			
		||||
		{9FF2A8A6-48D0-4D8A-9EAD-1905174291CC} = {9EB46BB6-D4EA-4B06-95EE-6C971E653030}
 | 
			
		||||
		{51313113-7BB8-494E-9C24-6787BECE39BB} = {BB9C2A85-7A8A-4CF9-BF44-34DE9848EC15}
 | 
			
		||||
		{79E7042F-F9E3-4D87-BFA9-4B7DD9736735} = {9EB46BB6-D4EA-4B06-95EE-6C971E653030}
 | 
			
		||||
		{000C3C62-345E-451C-8CEE-6F2C6A087116} = {9EB46BB6-D4EA-4B06-95EE-6C971E653030}
 | 
			
		||||
		{5D7BE567-2345-46C8-9F54-DDC1DA96D198} = {000C3C62-345E-451C-8CEE-6F2C6A087116}
 | 
			
		||||
		{5CF1B3EC-84E2-484A-8DFC-2ECD2EE18E2F} = {000C3C62-345E-451C-8CEE-6F2C6A087116}
 | 
			
		||||
		{CD0F211A-F65B-4026-9750-68AC3C70D012} = {000C3C62-345E-451C-8CEE-6F2C6A087116}
 | 
			
		||||
		{5CD79F91-7182-4A9D-9BEF-4DF410C782D2} = {000C3C62-345E-451C-8CEE-6F2C6A087116}
 | 
			
		||||
		{BB9C2A85-7A8A-4CF9-BF44-34DE9848EC15} = {9EB46BB6-D4EA-4B06-95EE-6C971E653030}
 | 
			
		||||
		{2057E5BE-FACA-4D44-A2BA-E1F864A8DAFF} = {CC8D0880-B73E-4DFC-9052-86504728708E}
 | 
			
		||||
		{A723D4D7-B796-4D97-BA68-95E5696C9559} = {CC8D0880-B73E-4DFC-9052-86504728708E}
 | 
			
		||||
		{D9944D52-81B4-4DBC-8C3B-2A334CCBD4F6} = {CC8D0880-B73E-4DFC-9052-86504728708E}
 | 
			
		||||
		{C5C89B2F-D9C9-4A67-A9BE-4E5F0B42162F} = {CC8D0880-B73E-4DFC-9052-86504728708E}
 | 
			
		||||
		{7562C85D-267D-4718-8857-422B625C5BD0} = {CC8D0880-B73E-4DFC-9052-86504728708E}
 | 
			
		||||
		{A0E1EAC7-10DD-4520-973F-CC385BBFBD84} = {CC8D0880-B73E-4DFC-9052-86504728708E}
 | 
			
		||||
		{503529C4-842A-47F0-928B-C8DC5B0A367D} = {CC8D0880-B73E-4DFC-9052-86504728708E}
 | 
			
		||||
		{8A9F6586-3320-4F03-B3DA-09BF39AA90A0} = {CC8D0880-B73E-4DFC-9052-86504728708E}
 | 
			
		||||
		{84362F1D-E788-4646-B555-5A3629B55EFC} = {237C7BC5-7B07-40B5-AF42-CE2F8E0893C3}
 | 
			
		||||
		{681F774F-7B0B-450A-917C-1385E1847CA6} = {237C7BC5-7B07-40B5-AF42-CE2F8E0893C3}
 | 
			
		||||
		{637A662B-7B70-4CE8-8F5F-0A095B9D77EC} = {95008B83-0324-4A7C-80DE-2BBDDD1A9099}
 | 
			
		||||
		{C5519C51-0A0C-4317-A43D-FFBB6B344ACB} = {95008B83-0324-4A7C-80DE-2BBDDD1A9099}
 | 
			
		||||
		{2192217C-CF77-422E-9E63-DF4003ABF01A} = {358AE1E1-4DD6-4D1F-8935-1ACBB0111A01}
 | 
			
		||||
		{DE376FAD-1E99-4CEE-96B2-08B1D1F3D1DB} = {358AE1E1-4DD6-4D1F-8935-1ACBB0111A01}
 | 
			
		||||
		{ACAA3F0D-CE2B-49F4-883A-ED23ADD9E325} = {358AE1E1-4DD6-4D1F-8935-1ACBB0111A01}
 | 
			
		||||
		{0A891D8E-23B3-46AD-8D30-565EE5004F93} = {4E66C22C-0636-4949-BF6A-9E3BBE1550BA}
 | 
			
		||||
		{A712EAEE-94F2-4F01-8C1C-2EC802280DD7} = {4E66C22C-0636-4949-BF6A-9E3BBE1550BA}
 | 
			
		||||
		{5DA3D2BD-6768-4479-B52F-49E022EFF310} = {4E66C22C-0636-4949-BF6A-9E3BBE1550BA}
 | 
			
		||||
		{8DD5DF98-7FDE-4B49-8661-AEB44D923CFE} = {4E66C22C-0636-4949-BF6A-9E3BBE1550BA}
 | 
			
		||||
		{D6685A42-2712-417A-92C5-5EFF90B9FA94} = {4E66C22C-0636-4949-BF6A-9E3BBE1550BA}
 | 
			
		||||
		{0D17D801-6DAA-4FD1-9A99-F9F07FA6BA88} = {4E66C22C-0636-4949-BF6A-9E3BBE1550BA}
 | 
			
		||||
		{1A4D0B95-9FC5-4687-94E2-B6F86B5427F2} = {DFB97103-FC3A-4DC3-A327-DA8C65BDA607}
 | 
			
		||||
		{69EA3D89-CB40-425A-8D70-5E4A33337BE5} = {DFB97103-FC3A-4DC3-A327-DA8C65BDA607}
 | 
			
		||||
		{90AB5C24-1AA3-4F58-9987-B307B92B5193} = {DFB97103-FC3A-4DC3-A327-DA8C65BDA607}
 | 
			
		||||
		{4C7A5A90-8292-413B-8848-419D9DCDE66F} = {DFB97103-FC3A-4DC3-A327-DA8C65BDA607}
 | 
			
		||||
		{D37EC028-EA46-4510-8261-6E780A906314} = {F0C9A8CB-231B-45E0-B91B-4FEF7EF47197}
 | 
			
		||||
		{C5F662EB-991F-438D-BF61-EF87E7371C04} = {F0C9A8CB-231B-45E0-B91B-4FEF7EF47197}
 | 
			
		||||
		{ECF3A7A3-2363-4A38-BC78-8FF9A8564603} = {E65490B8-D2E2-4693-B39C-15703B1EBFBB}
 | 
			
		||||
		{53C870AE-0249-4485-AE63-F8A16EA4BCB4} = {E65490B8-D2E2-4693-B39C-15703B1EBFBB}
 | 
			
		||||
		{E4B8B8E6-FAE7-43BA-9A51-33A3CD9FB825} = {E65490B8-D2E2-4693-B39C-15703B1EBFBB}
 | 
			
		||||
		{C102AB3A-377E-4753-AFA7-C13250D7DF00} = {E65490B8-D2E2-4693-B39C-15703B1EBFBB}
 | 
			
		||||
		{0DE1D254-0BD2-4D98-B939-5440BE5EB552} = {E65490B8-D2E2-4693-B39C-15703B1EBFBB}
 | 
			
		||||
		{C1911F84-6FDB-4DEC-8938-69B2E3901CC4} = {E65490B8-D2E2-4693-B39C-15703B1EBFBB}
 | 
			
		||||
		{A8E68E17-4EBF-4E4C-8272-B489329A68BF} = {E65490B8-D2E2-4693-B39C-15703B1EBFBB}
 | 
			
		||||
		{5FAFD46E-6B98-4C75-B1EB-085AA18F15FD} = {E65490B8-D2E2-4693-B39C-15703B1EBFBB}
 | 
			
		||||
		{AC4295F2-AB2F-4137-99EF-80FA5C83896B} = {E65490B8-D2E2-4693-B39C-15703B1EBFBB}
 | 
			
		||||
		{9DA9AED3-9572-4378-A2A6-4D792D67ADDC} = {E65490B8-D2E2-4693-B39C-15703B1EBFBB}
 | 
			
		||||
		{FD4969DB-3CCE-4CCF-BAE4-1BE8A3F40812} = {E65490B8-D2E2-4693-B39C-15703B1EBFBB}
 | 
			
		||||
		{0947E6C0-6371-4483-B0ED-191FB5073151} = {E65490B8-D2E2-4693-B39C-15703B1EBFBB}
 | 
			
		||||
	EndGlobalSection
 | 
			
		||||
	GlobalSection(ExtensibilityGlobals) = postSolution
 | 
			
		||||
		SolutionGuid = {C49B2D3E-6818-4E28-91B7-6E4E7E264BBB}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,17 +0,0 @@
 | 
			
		||||
<Project>
 | 
			
		||||
	<PropertyGroup>
 | 
			
		||||
		<Version>3.0.0.12</Version>
 | 
			
		||||
		<LangVersion>latest</LangVersion>
 | 
			
		||||
		<TargetFrameworks>net6.0;net7.0</TargetFrameworks>
 | 
			
		||||
		<Authors>Diego</Authors>
 | 
			
		||||
		<Product>ThingsGateway</Product>
 | 
			
		||||
		<Copyright>© 2023-present Diego</Copyright>
 | 
			
		||||
		<RepositoryUrl>https://gitee.com/diego2098/ThingsGateway</RepositoryUrl>
 | 
			
		||||
		<SignAssembly>True</SignAssembly>
 | 
			
		||||
		<DelaySign>False</DelaySign>
 | 
			
		||||
		<SatelliteResourceLanguages>zh-Hans</SatelliteResourceLanguages>
 | 
			
		||||
		<GenerateDocumentationFile>True</GenerateDocumentationFile>
 | 
			
		||||
	</PropertyGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
</Project>
 | 
			
		||||
@@ -1,270 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
using System.Diagnostics;
 | 
			
		||||
 | 
			
		||||
using ThingsGateway.Foundation.Core;
 | 
			
		||||
 | 
			
		||||
namespace ThingsGateway.Upgrade
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// StartCommand.txt 中配置启动主程序的指令,比如
 | 
			
		||||
    /// 如果是直接启动:ThingsGateway.Web.Entry.exe
 | 
			
		||||
    /// 如果是WindowsService:Net Start ThingsGateway
 | 
			
		||||
    /// 如果是其他部署(linux,pm2等等),填写对应的启动命令
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    internal class Program
 | 
			
		||||
    {
 | 
			
		||||
        static void Main(string[] args)
 | 
			
		||||
        {
 | 
			
		||||
            //Thread.Sleep(15000);
 | 
			
		||||
            Thread.Sleep(5000);
 | 
			
		||||
            var path = args[0];
 | 
			
		||||
            if (Directory.Exists(path))
 | 
			
		||||
            {
 | 
			
		||||
                var data = DirectoryUtility.GetDirectories(path);
 | 
			
		||||
                if (data.Length == 0)
 | 
			
		||||
                {
 | 
			
		||||
                    ConsleError($"目录{path}下不存在文件");
 | 
			
		||||
                }
 | 
			
		||||
                else
 | 
			
		||||
                {
 | 
			
		||||
                    var oldFolder = $"{AppContext.BaseDirectory}FileTemp/ThingsGatewayOld/{DateTimeExtensions.CurrentDateTime.ToFileDateTimeFormat()}";
 | 
			
		||||
                    try
 | 
			
		||||
                    {
 | 
			
		||||
 | 
			
		||||
                        //复制原文件,错误时可恢复
 | 
			
		||||
                        ConsleInfo($"正在备份原文件");
 | 
			
		||||
                        CopyDirectory(AppContext.BaseDirectory, oldFolder, new[] { "logs", "FileTemp" }, new[] { "ThingsGateway.Upgrade" });
 | 
			
		||||
                        ConsleInfo($"备份原文件成功");
 | 
			
		||||
 | 
			
		||||
                        //停止主程序进程
 | 
			
		||||
 | 
			
		||||
                        string filePath = $"{AppContext.BaseDirectory}ThingsGateway.Web.Entry";
 | 
			
		||||
                        // 查找正在运行的进程
 | 
			
		||||
                        var process = Process.GetProcesses().Where(p =>
 | 
			
		||||
                        {
 | 
			
		||||
                            try
 | 
			
		||||
                            {
 | 
			
		||||
                                if (p.ProcessName == "ThingsGateway.Web.Entry")
 | 
			
		||||
                                {
 | 
			
		||||
                                    return (Path.GetDirectoryName(p.MainModule.FileName) + Path.DirectorySeparatorChar).Equals(AppContext.BaseDirectory, StringComparison.OrdinalIgnoreCase);
 | 
			
		||||
                                    //return true;
 | 
			
		||||
                                }
 | 
			
		||||
                                return false;
 | 
			
		||||
                            }
 | 
			
		||||
                            catch (Exception)
 | 
			
		||||
                            {
 | 
			
		||||
                                return false; // 进程访问被拒绝或没有主模块
 | 
			
		||||
                            }
 | 
			
		||||
                        })?.FirstOrDefault();
 | 
			
		||||
                        if (process != null)
 | 
			
		||||
                        {
 | 
			
		||||
                            var exit = process.WaitForExit(300000);
 | 
			
		||||
                            if (!exit)
 | 
			
		||||
                            {
 | 
			
		||||
                                ConsleError("无法终止主程序,更新过程停止");
 | 
			
		||||
 | 
			
		||||
                                Recovery(oldFolder);
 | 
			
		||||
 | 
			
		||||
                                Directory.Delete(oldFolder, true);
 | 
			
		||||
                                Directory.Delete(path, true);
 | 
			
		||||
                                return;
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        //程序已退出
 | 
			
		||||
 | 
			
		||||
                        //复制文件到主程序文件夹
 | 
			
		||||
                        ConsleInfo($"正在复制文件到主程序文件夹");
 | 
			
		||||
                        CopyDirectory(path, AppContext.BaseDirectory, new[] { "logs", "FileTemp" }, new[] { "ThingsGateway.Upgrade" });
 | 
			
		||||
                        ConsleInfo($"复制文件到主程序文件夹成功");
 | 
			
		||||
 | 
			
		||||
                        //尝试启动
 | 
			
		||||
                        var dataString = FileUtil.ReadFile($"{AppContext.BaseDirectory}StartCommand.txt");//读取文件
 | 
			
		||||
                        var startCommand = dataString;
 | 
			
		||||
                        StartCommand(startCommand);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
                        Thread.Sleep(5000);
 | 
			
		||||
                        // 查找正在运行的进程
 | 
			
		||||
                        var process1 = Process.GetProcesses().Where(p =>
 | 
			
		||||
                        {
 | 
			
		||||
                            try
 | 
			
		||||
                            {
 | 
			
		||||
                                if (p.ProcessName == "ThingsGateway.Web.Entry")
 | 
			
		||||
                                {
 | 
			
		||||
                                    return (Path.GetDirectoryName(p.MainModule.FileName) + Path.DirectorySeparatorChar).Equals(AppContext.BaseDirectory, StringComparison.OrdinalIgnoreCase);
 | 
			
		||||
                                    //return true;
 | 
			
		||||
                                }
 | 
			
		||||
                                return false;
 | 
			
		||||
                            }
 | 
			
		||||
                            catch (Exception)
 | 
			
		||||
                            {
 | 
			
		||||
                                return false; // 进程访问被拒绝或没有主模块
 | 
			
		||||
                            }
 | 
			
		||||
                        })?.FirstOrDefault();
 | 
			
		||||
                        if (process1 != null)
 | 
			
		||||
                        {
 | 
			
		||||
                            //代表启动正常
 | 
			
		||||
                            ConsleInfo("更新成功");
 | 
			
		||||
                            Directory.Delete(oldFolder, true);
 | 
			
		||||
                            Directory.Delete(path, true);
 | 
			
		||||
                            return;
 | 
			
		||||
                        }
 | 
			
		||||
                        else
 | 
			
		||||
                        {
 | 
			
		||||
                            //恢复原文件
 | 
			
		||||
                            Recovery(oldFolder);
 | 
			
		||||
                            Directory.Delete(oldFolder, true);
 | 
			
		||||
 | 
			
		||||
                            Directory.Delete(path, true);
 | 
			
		||||
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                    }
 | 
			
		||||
                    catch (Exception ex)
 | 
			
		||||
                    {
 | 
			
		||||
                        ConsleError(ex.ToString());
 | 
			
		||||
 | 
			
		||||
                        //恢复原文件
 | 
			
		||||
                        Recovery(oldFolder);
 | 
			
		||||
                        Directory.Delete(oldFolder, true);
 | 
			
		||||
                        Directory.Delete(path, true);
 | 
			
		||||
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                ConsleError($"不存在目录{path}");
 | 
			
		||||
                //尝试启动
 | 
			
		||||
                var dataString1 = FileUtil.ReadFile($"{AppContext.BaseDirectory}StartCommand.txt");//读取文件
 | 
			
		||||
                var startCommand1 = dataString1;
 | 
			
		||||
                StartCommand(startCommand1);
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            Thread.Sleep(10000);
 | 
			
		||||
            Console.ReadLine();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        private static void Recovery(string oldFolder)
 | 
			
		||||
        {
 | 
			
		||||
            ConsleInfo("更新失败,恢复原文件");
 | 
			
		||||
            CopyDirectory(oldFolder, AppContext.BaseDirectory, new[] { "logs", "FileTemp" }, new[] { "ThingsGateway.Upgrade" });
 | 
			
		||||
            //尝试启动
 | 
			
		||||
            var dataString1 = FileUtil.ReadFile($"{AppContext.BaseDirectory}StartCommand.txt");//读取文件
 | 
			
		||||
            var startCommand1 = dataString1;
 | 
			
		||||
            StartCommand(startCommand1);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        /// <summary>
 | 
			
		||||
        /// 复制文件夹及文件
 | 
			
		||||
        /// </summary>
 | 
			
		||||
        /// <param name="sourceFolder">原文件路径</param>
 | 
			
		||||
        /// <param name="destFolder">目标文件路径</param>
 | 
			
		||||
        /// <param name="ignoreFolder">忽略文件夹</param>
 | 
			
		||||
        /// <param name="ignoreFiles">忽略文件</param>
 | 
			
		||||
        /// <returns></returns>
 | 
			
		||||
        public static void CopyDirectory(string sourceFolder, string destFolder, string[] ignoreFolder, string[] ignoreFiles)
 | 
			
		||||
        {
 | 
			
		||||
            //如果目标路径不存在,则创建目标路径
 | 
			
		||||
            if (!Directory.Exists(destFolder))
 | 
			
		||||
            {
 | 
			
		||||
                Directory.CreateDirectory(destFolder);
 | 
			
		||||
            }
 | 
			
		||||
            //得到原文件根目录下的所有文件
 | 
			
		||||
            var files = Directory.GetFiles(sourceFolder);
 | 
			
		||||
            foreach (var file in files)
 | 
			
		||||
            {
 | 
			
		||||
                if (ignoreFiles.Contains(Path.GetFileNameWithoutExtension(file)))
 | 
			
		||||
                { continue; }
 | 
			
		||||
                var name = Path.GetFileName(file);
 | 
			
		||||
                var dest = Path.Combine(destFolder, name);
 | 
			
		||||
                File.Copy(file, dest, true);//复制文件
 | 
			
		||||
            }
 | 
			
		||||
            //得到原文件根目录下的所有文件夹
 | 
			
		||||
            var folders = Directory.GetDirectories(sourceFolder);
 | 
			
		||||
            foreach (var folder in folders)
 | 
			
		||||
            {
 | 
			
		||||
                if (ignoreFolder.Contains(Path.GetFileName(folder)))
 | 
			
		||||
                { continue; }
 | 
			
		||||
                var name = Path.GetFileName(folder);
 | 
			
		||||
                var dest = Path.Combine(destFolder, name);
 | 
			
		||||
                CopyDirectory(folder, dest, ignoreFolder, ignoreFiles);//构建目标路径,递归复制文件
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        public static void ConsleInfo(string str)
 | 
			
		||||
        {
 | 
			
		||||
            Console.ForegroundColor = ConsoleColor.Green;
 | 
			
		||||
            Console.WriteLine(str);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        public static void ConsleError(string str)
 | 
			
		||||
        {
 | 
			
		||||
            Console.ForegroundColor = ConsoleColor.Red;
 | 
			
		||||
            Console.WriteLine(str);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        public static void StartCommand(string command)
 | 
			
		||||
        {
 | 
			
		||||
            // 创建进程对象
 | 
			
		||||
            Process process = new Process();
 | 
			
		||||
            // 设置进程启动信息
 | 
			
		||||
            process.StartInfo.FileName = GetCommandInterpreter();  // 获取当前平台的命令解释器
 | 
			
		||||
            process.StartInfo.Arguments = GetCommandArguments(command);  // 获取命令行参数
 | 
			
		||||
            process.StartInfo.UseShellExecute = true;
 | 
			
		||||
            process.StartInfo.WorkingDirectory = AppContext.BaseDirectory;
 | 
			
		||||
            // 启动进程
 | 
			
		||||
            process.Start();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // 获取当前平台的命令解释器
 | 
			
		||||
        static string GetCommandInterpreter()
 | 
			
		||||
        {
 | 
			
		||||
            if (System.Runtime.InteropServices.RuntimeInformation.IsOSPlatform(System.Runtime.InteropServices.OSPlatform.Windows))
 | 
			
		||||
            {
 | 
			
		||||
                return "cmd.exe";
 | 
			
		||||
            }
 | 
			
		||||
            else if (System.Runtime.InteropServices.RuntimeInformation.IsOSPlatform(System.Runtime.InteropServices.OSPlatform.Linux))
 | 
			
		||||
            {
 | 
			
		||||
                return "/bin/bash";
 | 
			
		||||
            }
 | 
			
		||||
            else if (System.Runtime.InteropServices.RuntimeInformation.IsOSPlatform(System.Runtime.InteropServices.OSPlatform.OSX))
 | 
			
		||||
            {
 | 
			
		||||
                return "/bin/bash";
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            throw new PlatformNotSupportedException("该平台不支持执行命令行。");
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // 获取命令行参数
 | 
			
		||||
        static string GetCommandArguments(string command)
 | 
			
		||||
        {
 | 
			
		||||
            if (System.Runtime.InteropServices.RuntimeInformation.IsOSPlatform(System.Runtime.InteropServices.OSPlatform.Windows))
 | 
			
		||||
            {
 | 
			
		||||
                return $"/c {command}";
 | 
			
		||||
            }
 | 
			
		||||
            else  // Linux 或 macOS
 | 
			
		||||
            {
 | 
			
		||||
                return $"-c \"{command}\"";
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -1,8 +0,0 @@
 | 
			
		||||
{
 | 
			
		||||
  "profiles": {
 | 
			
		||||
    "ThingsGateway.Upgrade": {
 | 
			
		||||
      "commandName": "Project",
 | 
			
		||||
      "commandLineArgs": "E:\\Tg\\ThingsGateway\\ThingsGateway-DEV\\framework\\Web\\ThingsGateway.Web.Entry\\bin\\Debug\\net6.0\\FileTemp\\tcp---127.0.0.1-7400-"
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1 +0,0 @@
 | 
			
		||||
Net Start ThingsGateway
 | 
			
		||||
@@ -1,32 +0,0 @@
 | 
			
		||||
<Project Sdk="Microsoft.NET.Sdk">
 | 
			
		||||
 | 
			
		||||
  <PropertyGroup>
 | 
			
		||||
    <OutputType>Exe</OutputType>
 | 
			
		||||
    <ImplicitUsings>enable</ImplicitUsings>
 | 
			
		||||
  </PropertyGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  <Target Name="PostBuild" AfterTargets="PostBuildEvent">
 | 
			
		||||
    <Exec Command=" set dir="$(SolutionDir)Web\ThingsGateway.Web.Entry\bin\$(Configuration)\$(TargetFramework)\"
 if not exist %25dir%25  md %25dir%25  
copy "$(TargetDir)*"  %25dir%25


" />
 | 
			
		||||
  </Target>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  <ItemGroup>
 | 
			
		||||
    <None Remove="StartCommand.txt" />
 | 
			
		||||
  </ItemGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  <ItemGroup>
 | 
			
		||||
    <Compile Include="..\..\Foundation\ThingsGateway.Foundation\Foundation\Utils\FileUtil.cs" Link="FileUtil.cs" />
 | 
			
		||||
    <Compile Include="..\..\Foundation\ThingsGateway.Foundation\TouchSocket\Core\Extensions\DateTimeExtensions.cs" Link="DateTimeExtensions.cs" />
 | 
			
		||||
    <Compile Include="..\..\Foundation\ThingsGateway.Foundation\TouchSocket\Core\IO\DirectoryUtility.cs" Link="DirectoryUtility.cs" />
 | 
			
		||||
  </ItemGroup>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  <ItemGroup>
 | 
			
		||||
    <Content Include="StartCommand.txt">
 | 
			
		||||
      <CopyToOutputDirectory>Always</CopyToOutputDirectory>
 | 
			
		||||
    </Content>
 | 
			
		||||
  </ItemGroup>
 | 
			
		||||
 | 
			
		||||
</Project>
 | 
			
		||||
@@ -1,26 +0,0 @@
 | 
			
		||||
@*
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
*@
 | 
			
		||||
 | 
			
		||||
@namespace ThingsGateway.UpgradeManger
 | 
			
		||||
 | 
			
		||||
<Router AppAssembly="@typeof(App).Assembly">
 | 
			
		||||
    <Found Context="routeData">
 | 
			
		||||
        <RouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)" />
 | 
			
		||||
        <FocusOnNavigate RouteData="@routeData" Selector="h1" />
 | 
			
		||||
    </Found>
 | 
			
		||||
    <NotFound>
 | 
			
		||||
        <LayoutView Layout="@typeof(MainLayout)">
 | 
			
		||||
            <p role="alert">Sorry, there's nothing at this address.</p>
 | 
			
		||||
        </LayoutView>
 | 
			
		||||
    </NotFound>
 | 
			
		||||
</Router>
 | 
			
		||||
 | 
			
		||||
@@ -1,23 +0,0 @@
 | 
			
		||||
#region copyright
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
//  此代码版权声明为全文件覆盖,如有原作者特别声明,会在下方手动补充
 | 
			
		||||
//  此代码版权(除特别声明外的代码)归作者本人Diego所有
 | 
			
		||||
//  源代码使用协议遵循本仓库的开源协议及附加协议
 | 
			
		||||
//  Gitee源代码仓库:https://gitee.com/diego2098/ThingsGateway
 | 
			
		||||
//  Github源代码仓库:https://github.com/kimdiego2098/ThingsGateway
 | 
			
		||||
//  使用文档:https://diego2098.gitee.io/thingsgateway-docs/
 | 
			
		||||
//  QQ群:605534569
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
#endregion
 | 
			
		||||
 | 
			
		||||
global using System;
 | 
			
		||||
global using System.Collections.Generic;
 | 
			
		||||
global using System.IO;
 | 
			
		||||
global using System.Linq;
 | 
			
		||||
global using System.Threading;
 | 
			
		||||
global using System.Threading.Tasks;
 | 
			
		||||
 | 
			
		||||
global using ThingsGateway.Components;
 | 
			
		||||
global using ThingsGateway.Foundation.Core;
 | 
			
		||||
global using ThingsGateway.Foundation.Rpc;
 | 
			
		||||
global using ThingsGateway.Foundation.Sockets;
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user