Compare commits
	
		
			754 Commits
		
	
	
		
			oldLinuxAc
			...
			2.1
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 1e6f806563 | ||
|  | 08ed93121b | ||
|  | 840cc57fdc | ||
|  | e46b0d22e7 | ||
|  | 085b0c1393 | ||
|  | e257ca0571 | ||
|  | 29e062ba5d | ||
|  | 2b9cd5508d | ||
|  | 79a591180d | ||
|  | 1e711eecbb | ||
|  | 7da01597e6 | ||
|  | 2e4bd4b1db | ||
|  | f3f936e401 | ||
|  | 90ed08c779 | ||
|  | c2874e74c8 | ||
|  | 3a11c9832f | ||
|  | 0f6ffb559b | ||
|  | 4769d7dc51 | ||
|  | 4fc2e98313 | ||
|  | 09a48c9c5a | ||
|  | ac245e71e9 | ||
|  | e5d2c711ce | ||
|  | de5e7380c4 | ||
|  | 3e87c90c67 | ||
|  | 4a97872cdd | ||
|  | 19e67f5a18 | ||
|  | 3e7c01f7f9 | ||
|  | 0bda00e43d | ||
|  | fe18faa863 | ||
|  | 808c715710 | ||
|  | 16d458bb69 | ||
|  | 22a7e2336e | ||
|  | f773cb5ecc | ||
|  | d1b6891ca7 | ||
|  | a9e65975c5 | ||
|  | c1377255b5 | ||
|  | cce8fa4869 | ||
|  | dda1018901 | ||
|  | b44d02dbd9 | ||
|  | cfded58563 | ||
|  | 8642f4291b | ||
|  | e83ff8d69d | ||
|  | 16edf04f1d | ||
|  | cbaa5cef46 | ||
|  | da75dc79a7 | ||
|  | 9965be5a76 | ||
|  | 9d6f3111da | ||
|  | 9265ab3ac3 | ||
|  | 34e70394e9 | ||
|  | 3a750018e2 | ||
|  | 390eea5c3d | ||
|  | dbf94b4c17 | ||
|  | bc8b2de151 | ||
|  | b73408043e | ||
|  | 2ba6e4fa7c | ||
|  | 9bbd7fd57d | ||
|  | 0ec3cb104a | ||
|  | b1f1d6701e | ||
|  | d54b447bf7 | ||
|  | 6ca7c5fb34 | ||
|  | ca9ad5efbd | ||
|  | b5154cbbe9 | ||
|  | a3ad88b9e6 | ||
|  | a4b9bf1fb9 | ||
|  | a95684d864 | ||
|  | 6057e8a3f8 | ||
|  | 4fbbfb8d10 | ||
|  | 2daafa2c4a | ||
|  | 846489244e | ||
|  | ec33b46848 | ||
|  | e067f09958 | ||
|  | 8cbfcbb104 | ||
|  | 2b5cf494db | ||
|  | 39158bd7ae | ||
|  | 95a695ee88 | ||
|  | a860732ed2 | ||
|  | 5d391a9fc4 | ||
|  | 148cd249cd | ||
|  | 57e643333e | ||
|  | 5794fab7da | ||
|  | 9955e7bcff | ||
|  | 86b252916c | ||
|  | e81a8fdc5f | ||
|  | 92cf8c36bd | ||
|  | de07a4f3cd | ||
|  | 320a530bbc | ||
|  | e945ad9fe8 | ||
|  | be3c242a7d | ||
|  | f86712a05a | ||
|  | 777ca4c016 | ||
|  | f715f48a7b | ||
|  | 315dd77a5a | ||
|  | 82f7f56590 | ||
|  | 8e594c7c7f | ||
|  | e99c77a97f | ||
|  | a80fc629a2 | ||
|  | 8b644d4fec | ||
|  | 17196ac31d | ||
|  | 632bb3e6d4 | ||
|  | def1c59e4c | ||
|  | e90e56c426 | ||
|  | 5f1676e56b | ||
|  | 72230c5a60 | ||
|  | fad5eca5b9 | ||
|  | 8054f088b9 | ||
|  | 544b8b7b25 | ||
|  | 4d6f54f54c | ||
|  | 2004f4f882 | ||
|  | 08e33ee20e | ||
|  | 330973ac30 | ||
|  | f78c755f12 | ||
|  | bbc2cdfa30 | ||
|  | d566285098 | ||
|  | 4c7b48588f | ||
|  | 22313a517f | ||
|  | 0c7a4911d2 | ||
|  | f11bccc868 | ||
|  | c62e90f0a1 | ||
|  | 170319f871 | ||
|  | c6dd782150 | ||
|  | cc7740bafb | ||
|  | 1626fe96b1 | ||
|  | efbca46af9 | ||
|  | 548f44b1a7 | ||
|  | 2664be7ec4 | ||
|  | bd3d0dd006 | ||
|  | a83e218679 | ||
|  | 8ee34cd42b | ||
|  | 1ad57ee70c | ||
|  | c9ab76c19c | ||
|  | 217655c276 | ||
|  | 57d1041f87 | ||
|  | b56bf7b13f | ||
|  | 3a137213a7 | ||
|  | 90cb224c9a | ||
|  | 69ad1baa7d | ||
|  | 6d2dee2664 | ||
|  | ace11fd77a | ||
|  | afa6f62daa | ||
|  | 6c8f9affc2 | ||
|  | fa0acb6a1a | ||
|  | 4fa7d26191 | ||
|  | e22f489038 | ||
|  | 11100e3fda | ||
|  | 047eac18a7 | ||
|  | 32b5354a15 | ||
|  | 0d085ba708 | ||
|  | d184f0eec3 | ||
|  | 675d502f73 | ||
|  | 4268004fcf | ||
|  | aabc5540d0 | ||
|  | 68806c6366 | ||
|  | 343fed3245 | ||
|  | c740457860 | ||
|  | 1d417c6075 | ||
|  | ec7d3d39bd | ||
|  | 2e98197376 | ||
|  | df17a86820 | ||
|  | 54c751e4c5 | ||
|  | 151d35459e | ||
|  | 1ae22ff249 | ||
|  | 983114af88 | ||
|  | 8a828bd087 | ||
|  | 76fc938604 | ||
|  | c08cb38dbc | ||
|  | f5dcb73e33 | ||
|  | 4108cf4df7 | ||
|  | f33c536adf | ||
|  | a66b36e142 | ||
|  | c9bff899ba | ||
|  | 82097b67f4 | ||
|  | 3a58e0e446 | ||
|  | 5ce61e1f98 | ||
|  | 5af198db75 | ||
|  | 1f6279cd46 | ||
|  | a07f1d5b24 | ||
|  | 26c0532fd5 | ||
|  | 9901bc2c8f | ||
|  | 715957feb2 | ||
|  | c7dc1e8b81 | ||
|  | 00798e5927 | ||
|  | 92c3fbd827 | ||
|  | a68837bf4b | ||
|  | 387b7f51d4 | ||
|  | 649a3515fc | ||
|  | fe8f8be1d8 | ||
|  | ea49e18f80 | ||
|  | cd9f9ad523 | ||
|  | bba494cdc0 | ||
|  | 393819bc94 | ||
|  | 49e3f72d89 | ||
|  | a266cdaeff | ||
|  | 43bf2de385 | ||
|  | 447dc23914 | ||
|  | 49aaaedbe9 | ||
|  | 7892a196ac | ||
|  | 7ab956fa4a | ||
|  | 17044d38bd | ||
|  | 0dc461c30b | ||
|  | ced5e06ff0 | ||
|  | 5e848cbc8c | ||
|  | f924604217 | ||
|  | efc59c0124 | ||
|  | 2eaa447155 | ||
|  | 3528424892 | ||
|  | 2eef7222d5 | ||
|  | 0d28a5b5f7 | ||
|  | e0e7e26234 | ||
|  | 2674563535 | ||
|  | a864fbac96 | ||
|  | 51c32a4350 | ||
|  | ec95f27ee6 | ||
|  | 15ce5aa938 | ||
|  | 39b353fb24 | ||
|  | 7a0c8aa977 | ||
|  | 664418d4dc | ||
|  | 381cf47abf | ||
|  | 1578c92a88 | ||
|  | 6827380e99 | ||
|  | 62f637790c | ||
|  | bf5311b5af | ||
|  | d0e0418ede | ||
|  | 243cbbfd4b | ||
|  | e7984d5a8e | ||
|  | eacce7adda | ||
|  | 622e8629ff | ||
|  | 50e368ee7a | ||
|  | 6766fe41fe | ||
|  | ba35968e8d | ||
|  | 4adfac1c02 | ||
|  | 437865b278 | ||
|  | 793b6c9004 | ||
|  | 96bd117622 | ||
|  | 8b2df76582 | ||
|  | 4196883bfb | ||
|  | 8585d10623 | ||
|  | c715b92ed0 | ||
|  | a3726399f1 | ||
|  | 8ce3efc7ee | ||
|  | 0bd3e1ac72 | ||
|  | c2d4e995e7 | ||
|  | 1b71fef8b4 | ||
|  | 074a4d525d | ||
|  | 36e19574e0 | ||
|  | ef9165b3a2 | ||
|  | 14cddbb210 | ||
|  | 2d14884454 | ||
|  | ba4eeffc77 | ||
|  | 59179b818e | ||
|  | dbcdc84b3b | ||
|  | c73dae361f | ||
|  | ec29371b41 | ||
|  | 9365f5937b | ||
|  | 971015d33a | ||
|  | 7a742c043b | ||
|  | c08a0cb0ed | ||
|  | b5926dec6f | ||
|  | 2fdcdb014f | ||
|  | bd70a6290e | ||
|  | 3afa96f1c5 | ||
|  | 5da12a8091 | ||
|  | 08eeff5604 | ||
|  | 169a946a03 | ||
|  | 94842ce0ef | ||
|  | 0fb7d5ed1b | ||
|  | 2021fd69ec | ||
|  | 813764a100 | ||
|  | cbd9330907 | ||
|  | a250cf4aef | ||
|  | 50bc3cd3ef | ||
|  | 4fb863cfa7 | ||
|  | 3a1bd1eed3 | ||
|  | bce3e429cf | ||
|  | b5baea184f | ||
|  | 42cbad4117 | ||
|  | 09f329db62 | ||
|  | 6b5f9d266d | ||
|  | 242d9b5e6e | ||
|  | 99b17e573c | ||
|  | 701edb91f1 | ||
|  | f0627db09f | ||
|  | 56a579e11b | ||
|  | 4427448eca | ||
|  | 203e2fcdd0 | ||
|  | f7fa92e6c1 | ||
|  | 425257a464 | ||
|  | 0acc07ebb3 | ||
|  | 030078a619 | ||
|  | 0e6ca4c188 | ||
|  | c7d5a1c928 | ||
|  | e636a4afcd | ||
|  | 8b76324ffc | ||
|  | 8fe1e55770 | ||
|  | caae694628 | ||
|  | 268e9d551a | ||
|  | 07137c2416 | ||
|  | 273b2a59c4 | ||
|  | 63364f4e72 | ||
|  | abc9622d53 | ||
|  | 9cd7e2f67b | ||
|  | 9277d3b5fb | ||
|  | 6f46e16be8 | ||
|  | 0b390e406a | ||
|  | f301e4654a | ||
|  | 62481899a1 | ||
|  | 869dfc8c06 | ||
|  | 939d456b9d | ||
|  | 9c70fb3caf | ||
|  | baf4a677dd | ||
|  | 4c4820f166 | ||
|  | 2749bfc40c | ||
|  | 704e0607eb | ||
|  | 401fbac63e | ||
|  | 7f1252a70a | ||
|  | 2df103a348 | ||
|  | 489bb44c92 | ||
|  | c606b6f00e | ||
|  | 0b4b38abe7 | ||
|  | 40b71fa983 | ||
|  | 6116db5147 | ||
|  | e979c6e1e2 | ||
|  | 8f9d042cdd | ||
|  | fcf030e693 | ||
|  | 31e6e01cad | ||
|  | 1aad0c85a4 | ||
|  | e6e16334b3 | ||
|  | 6544a61f72 | ||
|  | 8a285fddfa | ||
|  | c069f51e8d | ||
|  | c9a690fe8c | ||
|  | 847821c66c | ||
|  | b5387d4922 | ||
|  | b0a6807ea4 | ||
|  | 2e96074961 | ||
|  | f275a91a43 | ||
|  | 135392d245 | ||
|  | ff622bb9cd | ||
|  | 608c1317d7 | ||
|  | 9f4ef20dc1 | ||
|  | 73bd3cc28e | ||
|  | ea9eb04461 | ||
|  | c45833c252 | ||
|  | e260fc9790 | ||
|  | cc6ba2ff41 | ||
|  | bec74ddc99 | ||
|  | 7af7d11c8a | ||
|  | 8934d978fe | ||
|  | 8376e81532 | ||
|  | 6587d9ba2c | ||
|  | 2654036fd2 | ||
|  | f9226e7deb | ||
|  | 56ac8aece9 | ||
|  | 12737df530 | ||
|  | 7b9c835562 | ||
|  | cbb809db77 | ||
|  | 4f9085f0a2 | ||
|  | 389cf62150 | ||
|  | 44c367bf8f | ||
|  | efb0083161 | ||
|  | 16c1aba3e7 | ||
|  | 07ed8b9762 | ||
|  | fbd0a59a70 | ||
|  | a08fe53383 | ||
|  | e1b8c43cca | ||
|  | 60f20e64f3 | ||
|  | 6bb1109fe1 | ||
|  | 134f2059dd | ||
|  | ec93e5c9cc | ||
|  | 1487c21e1e | ||
|  | fd6b0c9458 | ||
|  | c41648748a | ||
|  | de4aef3a5c | ||
|  | d024d74529 | ||
|  | 6eb5a7ccc8 | ||
|  | 335fd338bd | ||
|  | d7ac59f257 | ||
|  | 1ff09a1a04 | ||
|  | ca54c2c099 | ||
|  | 76d209aa25 | ||
|  | 491a33421d | ||
|  | f1cbcf86e1 | ||
|  | e2e0f96d3e | ||
|  | 4be5d8d6e5 | ||
|  | 0365376ab0 | ||
|  | b9929566f6 | ||
|  | f463f586a3 | ||
|  | f6492256a8 | ||
|  | 6fe734f67e | ||
|  | 66d2c63a20 | ||
|  | 06f2faff35 | ||
|  | 57bd5e8149 | ||
|  | 11b9d07aef | ||
|  | b6f582d84e | ||
|  | b777919f52 | ||
|  | 6371a24d23 | ||
|  | d94cc70eff | ||
|  | 7d02f3682d | ||
|  | 51fb42db10 | ||
|  | a352059ddc | ||
|  | 0187b94768 | ||
|  | 163245401b | ||
|  | 7a0f28a227 | ||
|  | 0add4b4321 | ||
|  | 19fdcadbcd | ||
|  | 7b85adaddf | ||
|  | d8a0a2f80a | ||
|  | 8ec6ce7edb | ||
|  | 0eab8d0f3f | ||
|  | ae9a9534fc | ||
|  | b56dbcfdfd | ||
|  | 1c1003eb41 | ||
|  | 4bc6d88006 | ||
|  | fe6c701e44 | ||
|  | f08593881f | ||
|  | 9a59725504 | ||
|  | 30cf167e5b | ||
|  | 88b7b365e7 | ||
|  | 064d881c1e | ||
|  | c3f9d673bd | ||
|  | bbc2790ce0 | ||
|  | 64970b57ca | ||
|  | 4695dcaa0c | ||
|  | 2b11d1715b | ||
|  | e23f22d92d | ||
|  | c8492a4c47 | ||
|  | 962f1fcc1c | ||
|  | 2d6f4e4f2d | ||
|  | 6bc70ff4de | ||
|  | aaf4b539d5 | ||
|  | 83efb81966 | ||
|  | 03d0fc2ba4 | ||
|  | 41f128c2ad | ||
|  | bb09ef2a69 | ||
|  | 407b0ebf55 | ||
|  | 98d78d9a2e | ||
|  | 033ac0d1f6 | ||
|  | c9e9f60ed1 | ||
|  | 162c84e21c | ||
|  | 9f6d126484 | ||
|  | f1e27f1364 | ||
|  | 06c7a46d2a | ||
|  | 0877cc01f4 | ||
|  | 00e85357cf | ||
|  | cc72b3742f | ||
|  | dff20bd6e1 | ||
|  | 69b6f7d7d8 | ||
|  | 02a2800a4b | ||
|  | 9881df5117 | ||
|  | bc2f83c937 | ||
|  | fa78b4b295 | ||
|  | 95230292c3 | ||
|  | fdaf974009 | ||
|  | 26764c4cab | ||
|  | 52d0e8977c | ||
|  | 1d7df7814c | ||
|  | 1551fa8efa | ||
|  | 7e9a37e768 | ||
|  | 80bf6d77f1 | ||
|  | 2d7a2d6049 | ||
|  | 8363b41a68 | ||
|  | 9804678a30 | ||
|  | 1187c0aa5e | ||
|  | 36adbe387c | ||
|  | df2dcf7acd | ||
|  | b64fa374d7 | ||
|  | 3fbb492921 | ||
|  | 1b97e2015d | ||
|  | 3814986e41 | ||
|  | 037a4b523b | ||
|  | 522d493557 | ||
|  | 2ac06f7345 | ||
|  | 0745a7aa9a | ||
|  | 7e419aed8d | ||
|  | baa9565140 | ||
|  | 069dfb154a | ||
|  | eea9db7a0d | ||
|  | 485ed4d60b | ||
|  | 44056dda3a | ||
|  | 4de3e83ade | ||
|  | 73713eac69 | ||
|  | 6ed924655d | ||
|  | 7706262702 | ||
|  | d878d522d7 | ||
|  | e0843179cd | ||
|  | c6683db6e6 | ||
|  | 063b062913 | ||
|  | 3dd2521408 | ||
|  | f8e61d475a | ||
|  | e7cea4ff6e | ||
|  | 33d37ec0c2 | ||
|  | 512feaed81 | ||
|  | 93c6c1c20e | ||
|  | d2298f77b9 | ||
|  | 6979d9bc17 | ||
|  | d4135d527a | ||
|  | 91c678eb72 | ||
|  | 1ef2c7303f | ||
|  | 1b232d1113 | ||
|  | 78652b550b | ||
|  | 95d0965503 | ||
|  | d8f8815137 | ||
|  | 9bcf7076aa | ||
|  | ae71e2b4bd | ||
|  | f8934acbe4 | ||
|  | abf6d3751b | ||
|  | b77fa7dfd6 | ||
|  | 32a04edbf9 | ||
|  | 3ff281ffa3 | ||
|  | e07f1ae068 | ||
|  | 119635862b | ||
|  | e553b904c5 | ||
|  | 92db63d01d | ||
|  | 3584ff189b | ||
|  | e8614cbf14 | ||
|  | 399fc901fc | ||
|  | 524e234e6d | ||
|  | c14186fb4e | ||
|  | eeea4a3f39 | ||
|  | 567f91414f | ||
|  | e5710eb74c | ||
|  | 1733f142be | ||
|  | ccebca9748 | ||
|  | cb5d6ed771 | ||
|  | b9283c6778 | ||
|  | 5dc2c16cbe | ||
|  | 8d4f60b0fb | ||
|  | b9175be480 | ||
|  | 884bbe7363 | ||
|  | 98b9e6b659 | ||
|  | 1c4339f404 | ||
|  | 0ffc9964d0 | ||
|  | 18c9406900 | ||
|  | df498dd5bc | ||
|  | e7a3c7ebeb | ||
|  | 724aecd21c | ||
|  | b00a023f30 | ||
|  | 592510f0f3 | ||
|  | c96e566ebf | ||
|  | 78526b3560 | ||
|  | 04ec90ad61 | ||
|  | e69914a9a5 | ||
|  | c1c54f38cc | ||
|  | dad83c5c50 | ||
|  | b592e10126 | ||
|  | a235e43c07 | ||
|  | eb7185b9e3 | ||
|  | 776702f971 | ||
|  | 6fb9b11f1b | ||
|  | d3e2ab85b8 | ||
|  | 35a4dd659c | ||
|  | e0d2abbbd6 | ||
|  | 5db3429b2c | ||
|  | dd0c605ff4 | ||
|  | 54d41c1931 | ||
|  | 9ee66cf6ed | ||
|  | a34de035cb | ||
|  | 1dac61ea69 | ||
|  | 15c785d5fe | ||
|  | 7c7c609ea8 | ||
|  | f8b34e1d23 | ||
|  | 6295e7400d | ||
|  | 42a3df5c82 | ||
|  | 54a2a0cb35 | ||
|  | ec48f27792 | ||
|  | 15554ab08f | ||
|  | c042ab6b31 | ||
|  | 40641659a5 | ||
|  | f7fbc5a0c4 | ||
|  | 4a6e21d1ec | ||
|  | d6dcd265a4 | ||
|  | 8fe589fa75 | ||
|  | 656f10a1ca | ||
|  | f39d780a88 | ||
|  | 134e9c0b61 | ||
|  | f411eec5f4 | ||
|  | a996562a5e | ||
|  | 1ea3261b27 | ||
|  | a1bd7274f4 | ||
|  | 3206390c9c | ||
|  | 4ed20850ae | ||
|  | ef302205bc | ||
|  | e651d3348d | ||
|  | cc3cce6038 | ||
|  | 5b3e5242d1 | ||
|  | 60556ad4ec | ||
|  | 8686741410 | ||
|  | 199e0d3559 | ||
|  | 95b8969f5b | ||
|  | fc447e3ccf | ||
|  | 6b19e75ce7 | ||
|  | c3316a4745 | ||
|  | 250441056b | ||
|  | bdef8b2e8c | ||
|  | aef71ba48c | ||
|  | 86e5b9fcaa | ||
|  | 8d01e77a94 | ||
|  | d34248b868 | ||
|  | c15151a35d | ||
|  | b5609a7b0c | ||
|  | 4067de702c | ||
|  | 375f574e9e | ||
|  | ee59f45838 | ||
|  | ae5fafae76 | ||
|  | dc1dd977fc | ||
|  | f8b95382b0 | ||
|  | 2f655dce16 | ||
|  | c5138dbc9c | ||
|  | 8a17d5d017 | ||
|  | 089f62e6f8 | ||
|  | e4192e6064 | ||
|  | 646334ece5 | ||
|  | 2480a79450 | ||
|  | f173146d8f | ||
|  | 62f50304fe | ||
|  | c70d94e866 | ||
|  | dc006e7cbc | ||
|  | c5444f0df2 | ||
|  | c7d3e4022f | ||
|  | 151623e407 | ||
|  | 67393e88db | ||
|  | 2fe783e3ae | ||
|  | d87f4de4ff | ||
|  | b69f56ecb8 | ||
|  | b08fd3d60e | ||
|  | 7d26463639 | ||
|  | 4d9d4bdbee | ||
|  | 523ca37ee8 | ||
|  | 65c7b34ee0 | ||
|  | 2303861650 | ||
|  | 38b7ac892e | ||
|  | cfad948660 | ||
|  | ccf35fcd49 | ||
|  | d6e031e9e6 | ||
|  | 7b29d698dc | ||
|  | 989600e639 | ||
|  | 9fa4bacc74 | ||
|  | e02b3ad881 | ||
|  | cc4619220a | ||
|  | c71d86825c | ||
|  | 7648e9c655 | ||
|  | bb8931e7d4 | ||
|  | 695a67c2cb | ||
|  | 011ba37d2a | ||
|  | d73c8f49cc | ||
|  | 3ddf2f91ad | ||
|  | 809aea2c87 | ||
|  | 4e2fce147f | ||
|  | 26e01af696 | ||
|  | cb9359f9af | ||
|  | bad482b7f1 | ||
|  | a94f7b335b | ||
|  | 41b86509e0 | ||
|  | 27224fa8a1 | ||
|  | 9656aab27d | ||
|  | cbf68030d3 | ||
|  | 3e97481af2 | ||
|  | 705c3d7b76 | ||
|  | 2470e88d55 | ||
|  | f300770b29 | ||
|  | 63278938c8 | ||
|  | 5623c65763 | ||
|  | 5dca92ca5e | ||
|  | c1fd898d62 | ||
|  | 2435f589b9 | ||
|  | 6387629e7e | ||
|  | 5bde71b117 | ||
|  | b2eb7cc9c1 | ||
|  | 48df4390bd | ||
|  | 9920af9c59 | ||
|  | 681b1333d1 | ||
|  | 67b8955189 | ||
|  | d542503f55 | ||
|  | be5ed5a70b | ||
|  | f751fff8ca | ||
|  | 2d942fbac7 | ||
|  | 4bb4987937 | ||
|  | d371e3f7ac | ||
|  | 033fb6eff2 | ||
|  | 23819b77ee | ||
|  | fbcc2210b3 | ||
|  | 45b2e61631 | ||
|  | 4fceb8609f | ||
|  | 26c6b559a5 | ||
|  | 90e34029f3 | ||
|  | 6895f602ab | ||
|  | b1b8b613e6 | ||
|  | 40ba837f70 | ||
|  | c075ae5a9c | ||
|  | 8de49e5e94 | ||
|  | 7887b7f3cc | ||
|  | c81f162b90 | ||
|  | 35a5a78220 | ||
|  | 2bc0f64e34 | ||
|  | 19cb475e2d | ||
|  | ae4fc2d51a | ||
|  | 06e886a366 | ||
|  | 1497314a46 | ||
|  | e6998cb1f3 | ||
|  | 111782dffd | ||
|  | a86fbf83e8 | ||
|  | c9f6c17407 | ||
|  | c04037a4ea | ||
|  | d1adcebe4b | ||
|  | 3e5cbda861 | ||
|  | 8a393c6717 | ||
|  | f235bc0796 | ||
|  | 2549510eaa | ||
|  | 3fbcee60cb | ||
|  | 20cc73dca8 | ||
|  | 375fe6551c | ||
|  | f74524d947 | ||
|  | e1f26d2157 | ||
|  | 7d44bd7b65 | ||
|  | ab5f51a3b1 | ||
|  | d1e2e98099 | ||
|  | f94bbaad67 | ||
|  | 7143682062 | ||
|  | 17a087ed4a | ||
|  | 762138ca57 | ||
|  | 1fb46003fc | ||
|  | e1bbbece43 | ||
|  | e3e652d4e1 | ||
|  | b8a1fc31f3 | ||
|  | d5034c9971 | ||
|  | 215c8f8f3e | ||
|  | b2bf4bd0db | ||
|  | 1d2c4c8282 | ||
|  | 70fc09b21a | ||
|  | 9a529b8f5b | ||
|  | 4a2401b622 | ||
|  | 3feb502fec | ||
|  | e75c86ad58 | ||
|  | 9a95d39156 | ||
|  | cc48b50dcb | ||
|  | 32b1ac71a9 | ||
|  | bd98349dc4 | ||
|  | cf6602033d | ||
|  | 2959a878af | ||
|  | ea087fce53 | ||
|  | 9335c73003 | ||
|  | e11867a55e | ||
|  | c72a8d3950 | ||
|  | ddd8df3b26 | ||
|  | c491dc78ab | ||
|  | fd4fe245c6 | ||
|  | cbd8a17cc4 | ||
|  | 9356fc3559 | ||
|  | 243e627827 | ||
|  | bb408a31a2 | ||
|  | 3035360ce7 | ||
|  | cdaf5a17fa | ||
|  | f97e02e867 | ||
|  | 101aa6fc39 | ||
|  | 8e7a6af3ad | 
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,10 +1,12 @@ | ||||
| *.pyc | ||||
| *.pyo | ||||
| *.orig | ||||
| *~ | ||||
| *.swp | ||||
|  | ||||
| .DS_Store | ||||
| *_enterprise.* | ||||
| .settings/ | ||||
| .ipynb_checkpoints | ||||
|  | ||||
| # Debian buildings | ||||
| *.debhelper* | ||||
|   | ||||
							
								
								
									
										6
									
								
								TODO.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,6 @@ | ||||
| * Add "Scheduler" to manage pools, and posibbly other operations (Expect to release an version 1.7.1??) | ||||
| * Improve stats | ||||
| * Add App Virtualization support | ||||
| * Add "Meta Pools" | ||||
| * Manage to connect several UDS in "tree", so one UDS can be provider o another UDS | ||||
|  | ||||
							
								
								
									
										2
									
								
								actors/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -6,3 +6,5 @@ udsactor*.changes | ||||
| /udsactor_1.7.0.dsc | ||||
| /udsactor_1.7.0.tar.xz | ||||
| /udsactor*.rpm | ||||
| /udsactor_2.1.0_amd64.buildinfo | ||||
| linux/debian/files | ||||
|   | ||||
							
								
								
									
										2
									
								
								actors/linux/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1 +1 @@ | ||||
| /udsactor-opensuse-1.7.0.spec | ||||
| /udsactor-*[1-9].*.spec | ||||
|   | ||||
| @@ -1,16 +1,20 @@ | ||||
| #!/bin/bash | ||||
| 
 | ||||
| VERSION=1.7.0 | ||||
| VERSION=`cat ../../VERSION` | ||||
| RELEASE=1 | ||||
| 
 | ||||
| top=`pwd` | ||||
| 
 | ||||
| # Debian based | ||||
| dpkg-buildpackage -b | ||||
| 
 | ||||
| cat udsactor-template.spec |  | ||||
|   sed -e s/"version 1.7.0"/"version ${VERSION}"/g | | ||||
|   sed -e s/"version 0.0.0"/"version ${VERSION}"/g | | ||||
|   sed -e s/"release 1"/"release ${RELEASE}"/g > udsactor-$VERSION.spec | ||||
|    | ||||
| # Now fix dependencies for opensuse | ||||
| cat udsactor-template.spec |  | ||||
|   sed -e s/"version 0.0.0"/"version ${VERSION}"/g | | ||||
|   sed -e s/"name udsactor"/"name udsactor-opensuse"/g | | ||||
|   sed -e s/"PyQt4"/"python-qt4"/g | | ||||
|   sed -e s/"libXScrnSaver"/"libXss1"/g > udsactor-opensuse-$VERSION.spec | ||||
| @@ -27,4 +31,4 @@ for pkg in udsactor-$VERSION.spec udsactor-opensuse-$VERSION.spec; do | ||||
|     rpmbuild -v -bb --clean --buildroot=$top/rpm/BUILD/$pkg-root --target noarch $pkg 2>&1 | ||||
| done | ||||
| 
 | ||||
| #rm udsactor-$VERSION | ||||
| #rm udsactor-$VERSION | ||||
| @@ -1,4 +1,34 @@ | ||||
| udsactor (1.7.0) UNRELEASED; urgency=medium | ||||
| udsactor (2.1.0) stable; urgency=medium | ||||
|  | ||||
|   * Fixes for 2.1.0 release | ||||
|  | ||||
|  -- Adolfo Gómez García <agomez@virtualcable.es>  Tue, 19 Jan 2017 08:00:22 +0200 | ||||
|  | ||||
| udsactor (2.0.0) stable; urgency=medium | ||||
|  | ||||
|   * Upgrade for 2.0.0 | ||||
|  | ||||
|  -- Adolfo Gómez García <agomez@virtualcable.es>  Tue, 01 Mar 2016 03:39:21 +0100 | ||||
|  | ||||
| udsactor (1.9.1) stable; urgency=medium | ||||
|  | ||||
|   * Upgrade for 1.9.1 | ||||
|  | ||||
|  -- Adolfo Gómez García <agomez@virtualcable.es>  Tue, 01 Mar 2016 03:19:21 +0100 | ||||
|  | ||||
| udsactor (1.9.0) stable; urgency=medium | ||||
|  | ||||
|   * Upgrade for 1.9.0 (fixed package version) | ||||
|  | ||||
|  -- Adolfo Gómez García <agomez@virtualcable.es>  Tue, 05 May 2015 07:10:27 +0200 | ||||
|  | ||||
| udsactor (1.7.5) stable; urgency=medium | ||||
|  | ||||
|   * Upgrade for 1.7.5 | ||||
|  | ||||
|  -- Adolfo Gómez García <agomez@virtualcable.es>  Thu, 23 Apr 2015 06:08:53 +0200 | ||||
|  | ||||
| udsactor (1.7.0) stable; urgency=medium | ||||
|  | ||||
|   * Initial release. | ||||
|  | ||||
|   | ||||
| @@ -1,3 +0,0 @@ | ||||
| udsactor_1.7.0_all.deb admin optional | ||||
| udsactor-xrdp_1.7.0_all.deb x11 optional | ||||
| udsactor-nx_1.7.0_all.deb x11 optional | ||||
| @@ -5,6 +5,7 @@ | ||||
| set -e | ||||
| case "$1" in | ||||
| 	configure) | ||||
| 		/usr/bin/python2.7 -m compileall /usr/share/UDSActor > /dev/nul 2>&1 | ||||
| 		# If new "fresh" install or if configuration file has disappeared... | ||||
| 		if [ "$2" = "" ] || [ ! -f /etc/udsactor/udsactor.cfg ]; then | ||||
| 			db_get udsactor/host | ||||
|   | ||||
| @@ -8,5 +8,5 @@ Type=Application | ||||
| NoDisplay=true | ||||
| X-KDE-autostart-after=panel | ||||
| X-KDE-StartupNotify=false | ||||
| X-DBUS-StartupType=Unique | ||||
| X-KDE-UniqueApplet=true | ||||
| X-DBUS-StartupType=None | ||||
| X-KDE-UniqueApplet=false | ||||
|   | ||||
| @@ -1,69 +0,0 @@ | ||||
| %define _topdir %(echo $PWD)/rpm | ||||
| %define name udsactor | ||||
| %define version 1.7.0 | ||||
| %define release 1 | ||||
| %define buildroot %{_topdir}/%{name}-%{version}-%{release}-root | ||||
|  | ||||
| BuildRoot: %{buildroot}  | ||||
| Name: %{name} | ||||
| Version: %{version} | ||||
| Release: %{release} | ||||
| Summary: Actor for Universal Desktop Services (UDS) Broker | ||||
| License: BSD3 | ||||
| Group: Admin | ||||
| Requires: python-six python-requests PyQt4 libXScrnSaver | ||||
| Vendor: Virtual Cable S.L.U. | ||||
| URL: http://www.udsenterprise.com | ||||
| Provides: udsactor | ||||
|  | ||||
| %define _rpmdir ../ | ||||
| %define _rpmfilename %%{NAME}-%%{VERSION}-%%{RELEASE}.%%{ARCH}.rpm | ||||
|  | ||||
|  | ||||
| %install | ||||
| curdir=`pwd` | ||||
| cd ../.. | ||||
| make DESTDIR=$RPM_BUILD_ROOT DISTRO=rh install-udsactor | ||||
| cd $curdir | ||||
|  | ||||
| %clean | ||||
| rm -rf $RPM_BUILD_ROOT | ||||
| curdir=`pwd` | ||||
| cd ../.. | ||||
| make DESTDIR=$RPM_BUILD_ROOT DISTRO=rh clean | ||||
| cd $curdir | ||||
|  | ||||
|  | ||||
| %post | ||||
| systemctl enable udsactor.service > /dev/null 2>&1 | ||||
|  | ||||
| %preun | ||||
| systemctl disable udsactor.service > /dev/null 2>&1 | ||||
| systemctl stop udsactor.service > /dev/null 2>&1 | ||||
|  | ||||
| %postun | ||||
| # $1 == 0 on uninstall, == 1 on upgrade for preun and postun (just a reminder for me... :) ) | ||||
| if [ $1 -eq 0 ]; then | ||||
|     rm -rf /etc/udsactor | ||||
|     rm /var/log/udsactor.log | ||||
| fi | ||||
| # And, posibly, the .pyc leaved behind on /usr/share/UDSActor | ||||
| rm -rf /usr/share/UDSActor > /dev/null 2>&1 | ||||
|  | ||||
| %description | ||||
| This package provides the required components to allow this machine to work on an environment managed by UDS Broker. | ||||
|  | ||||
| %files | ||||
| %defattr(-,root,root) | ||||
| /etc/udsactor | ||||
| /etc/xdg/autostart/UDSActorTool.desktop | ||||
| /etc/init.d/udsactor | ||||
| /usr/bin/UDSActorTool-startup | ||||
| /usr/bin/udsactor | ||||
| /usr/bin/UDSActorTool | ||||
| /usr/sbin/UDSActorConfig | ||||
| /usr/sbin/UDSActorConfig-pkexec | ||||
| /usr/share/UDSActor/* | ||||
| /usr/share/applications/UDS_Actor_Configuration.desktop | ||||
| /usr/share/autostart/UDSActorTool.desktop | ||||
| /usr/share/polkit-1/actions/org.openuds.pkexec.UDSActorConfig.policy | ||||
| @@ -1,6 +1,6 @@ | ||||
| %define _topdir %(echo $PWD)/rpm | ||||
| %define name udsactor | ||||
| %define version 1.7.0 | ||||
| %define version 0.0.0 | ||||
| %define release 1 | ||||
| %define buildroot %{_topdir}/%{name}-%{version}-%{release}-root | ||||
|  | ||||
|   | ||||
| @@ -1,61 +0,0 @@ | ||||
| %define _topdir %(echo $PWD)/rpm | ||||
| %define name udsactor-xrdp | ||||
| %define version 1.7.0 | ||||
| %define release 1 | ||||
| %define buildroot %{_topdir}/%{name}-%{version}-%{release}-root | ||||
|  | ||||
| BuildRoot: %{buildroot}  | ||||
| Name: %{name} | ||||
| Version: %{version} | ||||
| Release: %{release} | ||||
| Summary: Glue between UDS Actor and XRDP | ||||
| License: BSD3 | ||||
| Group: Admin | ||||
| Requires: xrdp udsactor pam | ||||
| Vendor: Virtual Cable S.L.U. | ||||
| URL: http://www.udsenterprise.com | ||||
| Provides: udsactor-xrdp | ||||
|  | ||||
| %define _rpmdir ../ | ||||
| %define _rpmfilename %%{NAME}-%%{VERSION}-%%{RELEASE}.%%{ARCH}.rpm | ||||
|  | ||||
|  | ||||
| %install | ||||
| curdir=`pwd` | ||||
| cd ../.. | ||||
| make DESTDIR=$RPM_BUILD_ROOT DISTRO=rh install-udsactor-xrdp | ||||
| cd $curdir | ||||
|  | ||||
| %clean | ||||
| rm -rf $RPM_BUILD_ROOT | ||||
| curdir=`pwd` | ||||
| cd ../.. | ||||
| make DESTDIR=$RPM_BUILD_ROOT DISTRO=rh clean | ||||
| cd $curdir | ||||
|  | ||||
|  | ||||
| %post | ||||
| SESMANFILE=/etc/pam.d/xrdp-sesman | ||||
| TMPFILE=$(mktemp /tmp/sesman.XXXXX) | ||||
| grep -v uds $SESMANFILE > $TMPFILE | ||||
| echo >> $TMPFILE | ||||
| echo "# Added by udsactor-xrdp" >> $TMPFILE | ||||
| echo "session optional pam_exec.so /usr/bin/uds-sesman" >> $TMPFILE | ||||
| cp $TMPFILE $SESMANFILE | ||||
| rm $TMPFILE > /dev/null 2>&1 | ||||
|  | ||||
| %preun | ||||
|  | ||||
| %postun | ||||
| SESMANFILE=/etc/pam.d/xrdp-sesman | ||||
| TMPFILE=$(mktemp /tmp/sesman.XXXXX) | ||||
| grep -v uds $SESMANFILE > $TMPFILE | ||||
| cp $TMPFILE $SESMANFILE | ||||
| rm $TMPFILE > /dev/null 2>&1 | ||||
|  | ||||
| %description | ||||
| This package provides the required components to allow this machine to work on an environment managed by UDS Broker. | ||||
|  | ||||
| %files | ||||
| %defattr(-,root,root) | ||||
| /usr/bin/* | ||||
| @@ -33,10 +33,11 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import sys | ||||
| from PyQt4 import QtGui | ||||
| from PyQt4 import QtCore | ||||
| from PyQt4 import QtGui  # @UnresolvedImport | ||||
| from PyQt4 import QtCore  # @UnresolvedImport | ||||
| import pickle | ||||
| import time | ||||
| import datetime | ||||
| import signal | ||||
| from udsactor import ipc | ||||
| from udsactor import utils | ||||
| @@ -46,9 +47,12 @@ from udsactor import operations | ||||
| from about_dialog_ui import Ui_UDSAboutDialog | ||||
| from message_dialog_ui import Ui_UDSMessageDialog | ||||
| from udsactor.scriptThread import ScriptExecutorThread | ||||
| from udsactor import VERSION | ||||
|  | ||||
| trayIcon = None | ||||
|  | ||||
| doLogoff = False | ||||
|  | ||||
|  | ||||
| def sigTerm(sigNo, stackFrame): | ||||
|     if trayIcon: | ||||
| @@ -61,6 +65,7 @@ class UDSAboutDialog(QtGui.QDialog): | ||||
|         QtGui.QDialog.__init__(self, parent) | ||||
|         self.ui = Ui_UDSAboutDialog() | ||||
|         self.ui.setupUi(self) | ||||
|         self.ui.VersionLabel.setText("Version " + VERSION) | ||||
|  | ||||
|     def closeDialog(self): | ||||
|         self.hide() | ||||
| @@ -163,8 +168,9 @@ class UDSSystemTray(QtGui.QSystemTrayIcon): | ||||
|     def __init__(self, app_, parent=None): | ||||
|         self.app = app_ | ||||
|  | ||||
|         style = app.style() | ||||
|         icon = QtGui.QIcon(style.standardPixmap(QtGui.QStyle.SP_DesktopIcon)) | ||||
|         # style = app.style() | ||||
|         # icon = QtGui.QIcon(style.standardPixmap(QtGui.QStyle.SP_ComputerIcon)) | ||||
|         icon = QtGui.QIcon(':/images/img/uds.png') | ||||
|  | ||||
|         QtGui.QSystemTrayIcon.__init__(self, icon, parent) | ||||
|         self.menu = QtGui.QMenu(parent) | ||||
| @@ -172,13 +178,16 @@ class UDSSystemTray(QtGui.QSystemTrayIcon): | ||||
|         exitAction.triggered.connect(self.about) | ||||
|         self.setContextMenu(self.menu) | ||||
|         self.ipc = MessagesProcessor() | ||||
|         self.sessionStart = datetime.datetime.now() | ||||
|         self.maxIdleTime = None | ||||
|         self.timer = QtCore.QTimer() | ||||
|         self.timer.timeout.connect(self.checkIdle) | ||||
|         self.showIdleWarn = True | ||||
|         self.maxSessionTime = None | ||||
|         self.showMaxSessionWarn = True | ||||
|         self.timer = QtCore.QTimer() | ||||
|         self.timer.timeout.connect(self.checkTimers) | ||||
|  | ||||
|         if self.ipc.isAlive() is False: | ||||
|             raise Exception('no connection to service, exiting') | ||||
|             raise Exception('No connection to service, exiting.') | ||||
|  | ||||
|         self.stopped = False | ||||
|  | ||||
| @@ -197,36 +206,60 @@ class UDSSystemTray(QtGui.QSystemTrayIcon): | ||||
|         self.counter = 0 | ||||
|  | ||||
|         self.timer.start(5000)  # Launch idle checking every 5 seconds | ||||
|         self.graceTimerShots = 6  # Start counting for idle after 30 seconds after login, got on windows some "instant" logout because of idle timer not being reset?? | ||||
|  | ||||
|         self.ipc.start() | ||||
|         # If this is running, it's because he have logged in | ||||
|         self.ipc.sendLogin(operations.getCurrentUser()) | ||||
|  | ||||
|     def checkTimers(self): | ||||
|         self.checkIdle() | ||||
|         self.checkMaxSession() | ||||
|  | ||||
|     def checkMaxSession(self): | ||||
|         if self.maxSessionTime is None or self.maxSessionTime == 0: | ||||
|             logger.debug('Returning because maxSessionTime is cero') | ||||
|             return | ||||
|  | ||||
|         remainingTime = self.maxSessionTime - (datetime.datetime.now() - self.sessionStart).total_seconds() | ||||
|         logger.debug('Remaining time: {}'.format(remainingTime)) | ||||
|  | ||||
|         if self.showMaxSessionWarn is True and remainingTime < 300:  # With five minutes, show a warning message | ||||
|             self.showMaxSessionWarn = False | ||||
|             self.msgDlg.displayMessage('Your session will expire in less that 5 minutes. Please, save your work and disconnect.') | ||||
|             return | ||||
|  | ||||
|         if remainingTime <= 0: | ||||
|             logger.debug('Remaining time is less than cero, exiting') | ||||
|             self.quit() | ||||
|  | ||||
|     def checkIdle(self): | ||||
|         if self.maxIdleTime is None:  # No idle checl | ||||
|         if self.maxIdleTime is None:  # No idle check | ||||
|             return | ||||
|  | ||||
|         if self.graceTimerShots > 0: | ||||
|             self.graceTimerShots -= 1 | ||||
|             return | ||||
|  | ||||
|         idleTime = operations.getIdleDuration() | ||||
|         remainingTime = self.maxIdleTime - idleTime | ||||
|  | ||||
|         if remainingTime > 300:  # Reset show Warning dialog if we have more than 5 minutes left | ||||
|         if remainingTime > 120:  # Reset show Warning dialog if we have more than 5 minutes left | ||||
|             self.showIdleWarn = True | ||||
|  | ||||
|         logger.debug('User has been idle for: {}'.format(idleTime)) | ||||
|  | ||||
|         if remainingTime <= 0: | ||||
|             logger.info('User has been idle for too long, notifying Broker that service can be reclaimed') | ||||
|             self.quit() | ||||
|  | ||||
|         if self.showIdleWarn is True and remainingTime < 120:  # With two minutes, show a warning message | ||||
|             self.showIdleWarn = False | ||||
|             self.msgDlg.displayMessage("You have been idle for too long. The session will end if you don't resume operations") | ||||
|             logger.debug('Here') | ||||
|  | ||||
|         if remainingTime <= 0: | ||||
|             logger.info('User has been idle for too long, notifying Broker that service can be reclaimed') | ||||
|             self.quit(logoff=True) | ||||
|  | ||||
|     def displayMessage(self, message): | ||||
|         logger.debug('Displaying message') | ||||
|         self.msgDlg.displayMessage("You have been idle for too long. The session will end if you don't resume operations") | ||||
|         QtGui.QMessageBox.information(None, "UDS Actor", message) | ||||
|         self.msgDlg.displayMessage(message) | ||||
|  | ||||
|     def executeScript(self, script): | ||||
|         logger.debug('Executing script') | ||||
| @@ -235,13 +268,13 @@ class UDSSystemTray(QtGui.QSystemTrayIcon): | ||||
|  | ||||
|     def logoff(self): | ||||
|         self.counter += 1 | ||||
|         print("Loggof --", self.counter) | ||||
|         # print("Logofff --", self.counter) | ||||
|  | ||||
|     def information(self, info): | ||||
|         ''' | ||||
|         Invoked when received information from service | ||||
|         ''' | ||||
|         logger.debug('Got information message: {}'.format(info)) | ||||
|         logger.info('Got information message: {}'.format(info)) | ||||
|         if 'idle' in info: | ||||
|             idle = int(info['idle']) | ||||
|             operations.initIdleDuration(idle) | ||||
| @@ -250,23 +283,31 @@ class UDSSystemTray(QtGui.QSystemTrayIcon): | ||||
|         else: | ||||
|             self.maxIdleTime = None | ||||
|  | ||||
|         if 'maxSession' in info: | ||||
|             maxSession = int(info['maxSession']) | ||||
|             # operations.initMaxSession(maxSession) | ||||
|             self.maxSessionTime = maxSession | ||||
|             logger.debug('Set maxsession to {}'.format(maxSession)) | ||||
|  | ||||
|     def about(self): | ||||
|         self.aboutDlg.exec_() | ||||
|  | ||||
|     def quit(self): | ||||
|     def quit(self, logoff=False): | ||||
|         global doLogoff | ||||
|         logger.debug('Quit invoked') | ||||
|         if self.stopped is True: | ||||
|             return | ||||
|         self.stopped = True | ||||
|         try: | ||||
|             # If we close Client, send Logoff to Broker | ||||
|             self.ipc.sendLogout(operations.getCurrentUser()) | ||||
|             self.timer.stop() | ||||
|             self.ipc.stop() | ||||
|             operations.loggoff() | ||||
|         except Exception: | ||||
|             # May we have lost connection with server, simply exit in that case | ||||
|             pass | ||||
|         if self.stopped is False: | ||||
|             self.stopped = True | ||||
|             try: | ||||
|                 # If we close Client, send Logoff to Broker | ||||
|                 self.ipc.sendLogout(operations.getCurrentUser()) | ||||
|                 self.timer.stop() | ||||
|                 self.ipc.stop() | ||||
|             except Exception: | ||||
|                 # May we have lost connection with server, simply exit in that case | ||||
|                 pass | ||||
|  | ||||
|         doLogoff = logoff | ||||
|  | ||||
|         self.app.quit() | ||||
|  | ||||
| if __name__ == '__main__': | ||||
| @@ -282,9 +323,12 @@ if __name__ == '__main__': | ||||
|     try: | ||||
|         trayIcon = UDSSystemTray(app) | ||||
|     except Exception: | ||||
|         logger.error('UDS Service is not running. Tool stopped') | ||||
|         logger.error('UDS Service is not running, or it can\'t contact with UDS Server. User Tools stopped') | ||||
|         sys.exit(1) | ||||
|  | ||||
|     # Sets a default idle duration, but will not be used unless idle is notified from server | ||||
|     operations.initIdleDuration(3600 * 10) | ||||
|  | ||||
|     trayIcon.show() | ||||
|  | ||||
|     # Catch kill and logout user :) | ||||
| @@ -295,4 +339,12 @@ if __name__ == '__main__': | ||||
|     logger.debug('Exiting') | ||||
|     trayIcon.quit() | ||||
|  | ||||
|     if doLogoff: | ||||
|         try: | ||||
|             time.sleep(1) | ||||
|             operations.loggoff()  # Invoke log off | ||||
|         except Exception: | ||||
|             pass | ||||
|  | ||||
|  | ||||
|     sys.exit(res) | ||||
|   | ||||
| @@ -2,225 +2,177 @@ | ||||
|  | ||||
| # Resource object code | ||||
| # | ||||
| # Created: Fri Nov 21 12:40:02 2014 | ||||
| # Created: Mon Apr 27 22:05:02 2015 | ||||
| #      by: The Resource Compiler for PyQt (Qt v4.8.6) | ||||
| # | ||||
| # WARNING! All changes made in this file will be lost! | ||||
|  | ||||
| from PyQt4 import QtCore | ||||
|  | ||||
| qt_resource_data = "\ | ||||
| \x00\x00\x0c\xe0\ | ||||
| qt_resource_data = b"\ | ||||
| \x00\x00\x09\xd1\ | ||||
| \x89\ | ||||
| \x50\x4e\x47\x0d\x0a\x1a\x0a\x00\x00\x00\x0d\x49\x48\x44\x52\x00\ | ||||
| \x00\x00\x30\x00\x00\x00\x30\x08\x06\x00\x00\x00\x57\x02\xf9\x87\ | ||||
| \x00\x00\x00\x04\x67\x41\x4d\x41\x00\x00\xb1\x8f\x0b\xfc\x61\x05\ | ||||
| \x00\x00\x00\x20\x63\x48\x52\x4d\x00\x00\x7a\x26\x00\x00\x80\x84\ | ||||
| \x00\x00\xfa\x00\x00\x00\x80\xe8\x00\x00\x75\x30\x00\x00\xea\x60\ | ||||
| \x00\x00\x3a\x98\x00\x00\x17\x70\x9c\xba\x51\x3c\x00\x00\x00\x06\ | ||||
| \x62\x4b\x47\x44\x00\xff\x00\xff\x00\xff\xa0\xbd\xa7\x93\x00\x00\ | ||||
| \x00\x07\x74\x49\x4d\x45\x07\xde\x0b\x11\x04\x03\x27\xf2\xbb\x98\ | ||||
| \x60\x00\x00\x0b\xe4\x49\x44\x41\x54\x68\xde\xe5\x9a\x7b\x6c\x1c\ | ||||
| \xc7\x79\xc0\x7f\x33\xbb\x7b\xef\x3b\x8a\xa4\xf8\x96\x48\x49\xa4\ | ||||
| \x25\x59\x91\x28\xc9\x72\xf4\x96\x6d\xc5\x6d\x9a\xca\x71\x22\x03\ | ||||
| \xb2\xe3\x5a\x71\xd2\x04\xa8\xdd\xb8\x06\x52\xd8\x4e\xd5\x36\x29\ | ||||
| \x60\xd4\x70\x5c\x38\x69\x8a\x06\x4d\xd0\xba\x45\x03\xb8\xa8\x0b\ | ||||
| \x34\x45\x0d\xd7\xad\x9d\x1a\x4d\xdc\xd4\x8f\xc0\x55\xf4\xb2\xf5\ | ||||
| \x20\x65\x53\x2f\x4a\x22\x45\x4a\x24\xef\x8e\xf7\xda\xdd\x99\xe9\ | ||||
| \x1f\x47\x52\x77\xe4\x1d\x7d\x92\x6c\x14\x45\x3f\xe0\x70\xb7\x3b\ | ||||
| \xdf\x7e\xf3\xfd\xbe\x99\x6f\xf6\x9b\xdd\x13\xc6\x18\xc3\xff\x61\ | ||||
| \x91\xff\xdb\x0e\xfc\xbf\x07\xb0\xaf\x45\xd9\x18\x83\x31\x3e\xda\ | ||||
| \xb8\x18\xe3\xa1\x8d\x87\xd6\x2e\x06\x1f\x8c\xc2\xa0\xd1\x5a\xe1\ | ||||
| \xfb\x06\xcf\xd3\xf8\xbe\x41\x69\x83\xd6\x60\x4c\xf1\x03\x02\x21\ | ||||
| \x04\x96\xb4\xb1\x2c\x1b\xc7\x0e\x60\xdb\x01\x02\x4e\x88\x60\x20\ | ||||
| \x8c\x6d\x3b\x1f\x1d\x80\x31\x06\xa5\x33\x14\xdc\x21\xf2\xee\x30\ | ||||
| \xae\x37\x4a\xde\x1b\xa6\xe0\x0d\xe1\xf9\x13\x78\x6a\x02\xcf\x9f\ | ||||
| \x40\xeb\x49\x94\xce\x71\x69\x24\xc3\xc5\xa1\x3c\xa3\x97\x0b\x8c\ | ||||
| \x8d\x17\x48\x4f\x7a\x14\x5c\x83\xef\x1a\x3c\x05\x46\x0b\x40\x60\ | ||||
| \x59\x01\xc2\xa1\x18\xe1\x70\x82\x44\xb4\x81\x44\xbc\x81\xfa\xba\ | ||||
| \x16\xda\x5b\x96\xb1\xa8\xed\x26\x3a\x5a\x7b\x68\x6f\x5e\x8a\xe3\ | ||||
| \x04\x3f\x14\x40\x54\x4a\x62\x5f\x65\x98\xcc\xf5\x31\x96\x7a\x93\ | ||||
| \x54\xf6\x28\x79\x77\x10\xd7\xbb\x82\xaf\x92\x28\x95\x45\xe3\x21\ | ||||
| \x00\x03\x08\xc0\x75\x0d\xef\x1e\xf5\x38\xf4\x6e\x81\x74\xd2\xc5\ | ||||
| \x53\x36\x9e\x49\xa0\x09\x82\xb0\x40\x48\x04\x80\x50\x48\xe3\x23\ | ||||
| \x85\x42\x90\x47\xea\x2c\xc2\xa8\x62\xb0\x84\xc4\xb6\xc3\xc4\x63\ | ||||
| \x0d\x34\x37\x2e\x62\xf5\xf2\x2d\xec\xdc\xfa\x05\xba\xbb\xd6\xcc\ | ||||
| \x0b\x52\x06\xe0\x7a\x57\x18\x9d\x78\x8d\xf3\x97\xff\x9e\x4c\xee\ | ||||
| \x7d\x94\xce\x62\x8c\x87\xa1\xa8\x32\xed\x74\xe9\x6f\x63\x0c\x07\ | ||||
| \x0e\xba\xfc\xfc\xcd\x3c\x79\xd5\xc8\x84\xbd\x93\x94\xbd\x8d\x9c\ | ||||
| \x5c\x84\x21\x80\xc1\x9a\xd2\x06\x21\xc0\x8e\x26\xb0\x63\x31\x2c\ | ||||
| \x9d\xc1\xf6\x47\x08\xba\xfd\x04\x0b\x27\x08\x15\x8e\x13\xf0\x06\ | ||||
| \x90\x3a\x8d\x30\x3e\x96\x65\x13\x8f\x35\xb2\x75\xc3\x5d\xec\xdd\ | ||||
| \xbd\x8f\x96\xa6\xae\xf9\x01\x5c\xef\x32\xfd\x83\x4f\x32\x3c\xf6\ | ||||
| \x2f\x68\x53\x98\xea\xf2\xaa\xc3\x65\x17\x95\xb4\x69\x65\xf8\xd9\ | ||||
| \x7f\xe6\xf8\xc5\x91\x66\xce\x86\xbf\x45\x5a\xae\x9b\xd2\xf1\xa7\ | ||||
| \xbe\xa7\xe0\x9d\x20\x76\xbc\x09\x19\x49\x4c\x59\x90\x18\x21\x01\ | ||||
| \x89\x30\x1a\xa9\xd3\x04\xdd\x3e\xa2\xd9\xb7\x88\x65\x5e\x27\x9c\ | ||||
| \xdf\x8f\xd4\x93\x20\x24\x9b\xd7\xef\xe2\x91\x2f\x7d\x87\xf6\x96\ | ||||
| \x65\x73\x7c\x99\x59\x85\x86\xc6\x5e\x62\x78\xec\x45\xb4\x29\x5c\ | ||||
| \xcd\x81\x1a\x92\xc8\xb2\xa0\xad\xcd\x22\x19\xf9\x1c\x69\xb9\x0e\ | ||||
| \x81\x87\xc4\x43\x60\x8a\xce\x1b\x83\xb0\x03\x38\xf5\xed\xc8\x70\ | ||||
| \x7c\xea\xbc\x46\xe0\x23\x8d\x8b\x34\x79\x04\x1e\x5a\x86\xc9\x86\ | ||||
| \x36\x70\xb9\xe1\x51\xce\xb7\xfd\x90\xe1\xe6\x67\xf0\xec\x0e\x30\ | ||||
| \x9a\xfd\x47\x5e\xe3\x85\x97\x9e\x65\x3c\x39\x52\x19\xc0\xf7\x5d\ | ||||
| \x06\x87\xdf\x41\xa0\x6a\x72\xde\x94\xb4\x1b\x60\xe5\x4d\x36\x9b\ | ||||
| \xd7\xe6\xc0\xe8\xb9\x5a\xb6\x83\x5d\xdf\x8e\x0c\x84\x11\xa2\xba\ | ||||
| \x45\x81\x46\xe2\x03\x1a\xdf\x6e\x66\x22\xf1\x00\xc3\xcd\xcf\xe0\ | ||||
| \xdb\xed\x28\xe5\xf1\xce\xa1\x57\x39\xda\xff\x36\xb3\x53\x56\x02\ | ||||
| \x28\xed\x31\x30\x38\x8e\xeb\x6b\xaa\xf6\x51\x55\x04\x4e\x40\xb0\ | ||||
| \x67\xcb\x4b\xdc\xbe\xfc\xa7\x68\x23\xaf\xc2\x4b\x0b\x27\xd1\x8c\ | ||||
| \x0c\x46\xe6\x09\x89\x98\x75\x24\x8a\xf9\x25\x20\x15\xdf\xc5\xd8\ | ||||
| \x82\x2f\x63\x84\xcd\x78\x72\x84\xfd\x47\x5e\x23\x93\x4b\xcd\x05\ | ||||
| \xb0\xed\x20\x9e\xdb\x40\xdf\x29\x77\x2a\x1a\x57\x4d\x8b\x79\xba\ | ||||
| \x9d\x6e\xd3\x46\x62\x09\x8f\x3d\xb7\xbe\xc0\x8a\x96\x7e\xb4\xb6\ | ||||
| \x30\x46\x20\x9c\x10\x22\x14\x9d\xc9\x83\xea\xe3\x39\x77\x7c\x8b\ | ||||
| \xd7\xd8\x4c\x46\x7f\x05\x2d\xe3\x80\xe1\xf4\xe0\x31\xb2\xd9\x0a\ | ||||
| \x00\x96\xb4\x59\xde\xb5\x95\xfe\x01\xc5\xa9\x41\xaf\xcc\xa8\x99\ | ||||
| \xa7\xdb\xd2\x36\x65\x2c\x5a\x12\x17\x78\x70\xf3\xdf\xb0\xa8\x7e\ | ||||
| \x10\x83\x40\x3a\x21\x84\xbc\xa6\x7b\xe5\x9c\x5e\x7c\xab\x0d\x25\ | ||||
| \xe3\x60\x0c\x63\x13\x97\x70\xbd\xfc\x5c\x00\x80\x4d\x6b\xee\x65\ | ||||
| \x49\xfb\x56\x8e\x9c\xc8\x71\xe9\x8a\x2a\x5b\x32\x67\x0f\xb4\x98\ | ||||
| \xf5\x29\x6d\x59\xd5\xf1\x2e\x5f\xdd\xfe\x97\xc4\x82\x19\x0c\xf6\ | ||||
| \x9c\xf6\xea\x96\x2a\xeb\x18\x61\x81\x28\x06\xc1\x57\x2e\xba\x2c\ | ||||
| \xcf\x4a\x00\x82\x81\x08\xbb\x77\xfe\x21\xd1\xf0\x32\x0e\x1e\xcd\ | ||||
| \x92\xce\x28\x64\x89\xdd\x52\x98\xd2\x04\x2e\x3f\x2f\x50\xda\xa2\ | ||||
| \x77\xf1\x01\xf6\x6c\xf8\x07\x1c\x91\x46\x95\xf5\x37\x7d\xc5\x7c\ | ||||
| \x8b\x74\xb9\x08\x53\x40\xe8\x02\x08\x08\xd8\x41\xa4\x28\x2f\xdf\ | ||||
| \xca\x8e\x7a\x3a\x37\x73\xe7\xc6\x87\x48\x4d\x06\x39\x74\x3c\x87\ | ||||
| \xd2\xa6\x6a\x1e\x54\x9f\x5a\x02\x63\xe0\xd3\x9f\xf8\x57\x3e\xb3\ | ||||
| \xe2\x9f\x91\xc6\xc7\x98\xeb\xad\x19\x25\xb6\x1a\x47\x9a\x2c\x20\ | ||||
| \x48\xc4\x1b\x71\xec\xe0\x2c\x8d\x12\x71\xec\x20\xdb\xd7\x3f\xc8\ | ||||
| \x96\xde\xfb\x38\x3f\xac\x39\x72\x22\x8f\xaf\xae\xba\x5a\x09\xa4\ | ||||
| \xf2\x54\x92\xd8\x96\xcb\x3d\x6b\x7f\xc4\xa7\x96\xfc\x23\x52\x28\ | ||||
| \x8c\x29\xd5\x98\x3d\x9e\x25\xc7\x57\xab\xbe\x62\x1a\xfb\xe7\x91\ | ||||
| \xda\xc5\x20\xe8\x68\xed\x21\x1c\x8a\x55\x07\x00\x88\x47\x1b\xb9\ | ||||
| \xfb\xf6\x7d\x74\xb6\xae\xa5\xff\x74\x9e\x81\x73\xee\x4c\x29\x31\ | ||||
| \xdb\xf1\xf9\xc4\x20\x09\xdb\x93\xec\x59\xf5\xa7\x6c\x6a\x7b\x05\ | ||||
| \x65\xec\x59\x10\x55\x64\x26\x1a\x02\x81\xc1\xf1\x2e\x02\x2e\x52\ | ||||
| \x48\xda\x5b\x96\x11\x0a\x46\xe6\x07\x00\x68\x69\xec\x66\xf7\xce\ | ||||
| \x6f\x62\xcb\x04\x47\x4f\xe6\x18\x9b\x28\xe6\x83\xa9\xd0\x57\xa5\ | ||||
| \x73\xd3\xa2\xb1\x88\x05\xc6\xb8\x7f\xd5\xd3\xac\x5e\xf8\xd6\x14\ | ||||
| \x44\x6d\x04\x06\x03\xc6\xc7\xf1\xcf\x83\xf1\x08\x87\x62\xb4\xb7\ | ||||
| \x74\xe3\x38\x81\x0f\x07\x00\x58\xdd\x73\x27\xbb\x76\x3c\x86\xe7\ | ||||
| \x85\x78\xeb\x40\x86\x74\x46\x97\x25\xf5\xf4\x10\x97\x7e\x4f\xff\ | ||||
| \x2e\x3d\xd6\x46\x52\x1f\x1a\xe6\x6b\xb7\x7c\x9d\x3b\x16\xff\x98\ | ||||
| \x68\xc0\x23\x12\x90\x04\x2c\x31\xc7\x5e\xb9\x35\x89\x65\x72\x04\ | ||||
| \xbc\xb3\x80\x21\x1e\xab\x67\x61\x7d\x1b\x62\x56\x12\x57\x5d\xa4\ | ||||
| \x03\x4e\x84\xed\xeb\xf7\x32\x78\x69\x3f\xff\xfd\xde\x2b\x1c\x3c\ | ||||
| \x96\x65\x63\x6f\x84\x50\xf0\xaa\x81\x4a\x23\x50\x29\xa2\x1a\x58\ | ||||
| \x10\x1c\x66\xef\xea\x27\xd9\x3e\x79\x12\x3b\xfc\x04\x59\x3f\xca\ | ||||
| \x68\xda\xe7\x42\xd2\x30\x32\xa9\xb9\x90\x34\x8c\xe7\x4a\x43\x22\ | ||||
| \x90\x3a\x8d\xed\x5f\x42\x00\x89\x68\x3d\xf5\x75\xcd\x73\xac\x57\ | ||||
| \x05\xc8\xbb\xa7\x30\xe2\xbf\xd8\xd8\x1b\xe6\xe4\xd9\x10\xe7\x2e\ | ||||
| \x66\xa9\x8b\x5b\xf4\xae\x08\x81\x10\x73\x72\x60\x76\x99\x3d\x17\ | ||||
| \xc2\x22\x6c\xa7\x58\xd3\xf8\x3c\x8b\x9a\xa0\xb3\xf5\x77\x51\xa6\ | ||||
| \x81\x64\xde\x30\x9e\x35\x5c\x48\x1a\xde\x3e\xe3\xf3\xc6\x29\xc5\ | ||||
| \xc9\x51\x8d\xa7\xc1\x52\x63\xd8\xfe\x15\x10\x82\xc6\xfa\x76\x16\ | ||||
| \x36\x74\xcc\x0f\xe0\xf9\x59\xb2\x85\x3e\x32\xf9\x37\x48\xe7\xf6\ | ||||
| \x63\x4c\x9a\x05\x75\x9a\xad\x1b\x7a\xf9\xb7\xd7\x8f\x70\xfc\xfd\ | ||||
| \x3c\xb1\xa8\xa4\xa7\x33\x58\x5c\x2c\xaa\xc6\xbc\x72\x9b\x41\xe2\ | ||||
| \xa9\x1c\xe7\x46\xfe\x0e\x08\xb0\xa4\xed\x51\xda\x12\xf5\xb4\x25\ | ||||
| \xe0\xe6\x16\xc3\x8e\x6e\x8b\x87\xb7\x18\xfe\xbd\xdf\xe7\xfb\x6f\ | ||||
| \xf8\xa4\x86\x2e\x61\xe9\x2b\xd8\x96\xcd\xd2\xc5\xab\x88\x45\x17\ | ||||
| \xcc\x0f\x30\x30\xf8\x36\xbf\x38\xfa\x07\x2c\x5f\x9a\xa0\x3e\x11\ | ||||
| \x26\x99\x6b\xe2\x83\xd1\xf5\x1c\x1e\xdd\x8e\x8e\xbf\x84\x97\xfe\ | ||||
| \x33\x0e\xbc\x97\xa5\x2e\x66\xd1\xd4\x50\x4c\xc8\xda\xa6\x51\x29\ | ||||
| \x84\xc0\xd7\x39\xce\x8e\xfc\x08\x03\x74\x77\xfc\x1e\x96\x0c\x22\ | ||||
| \x84\x20\x64\x43\x6b\x42\xf0\xe5\x4f\x06\x68\x89\xc1\x93\xcf\x5f\ | ||||
| \x44\xea\x34\x4e\x28\xc4\x8a\xee\x5b\x11\x15\xca\xd9\x32\x00\x5f\ | ||||
| \x49\xde\x3c\xd0\xc7\x89\x01\x87\xce\x9e\xaf\x32\x90\xfa\x0a\x43\ | ||||
| \xc9\xa5\x14\xfc\x30\x76\xa4\x91\x80\x7b\x80\x7c\xfe\x27\x1c\x39\ | ||||
| \x91\x63\xcb\x2d\x51\x22\x21\x59\xb1\x14\xab\x65\x54\xb4\xce\x70\ | ||||
| \x6e\xe4\x6f\x89\x85\x57\xd0\xd6\x78\xef\x1c\xe7\x36\x76\xc2\xad\ | ||||
| \x2d\x49\xfa\xce\xb8\x84\x82\x0d\x2c\xeb\xea\xad\x68\xb7\x2c\xa5\ | ||||
| \x1b\xea\xda\x49\x44\x5b\x18\x1a\x99\xe0\xcd\xe3\x3d\x9c\x1d\x5f\ | ||||
| \x85\xa7\x22\x48\x21\xd0\xd6\x62\xbc\xf8\x37\x11\xce\x4d\x0c\x8d\ | ||||
| \xba\x1c\x3e\x9e\xc3\xf5\x4c\xcd\xe5\x77\x25\x50\xa5\xb3\x0c\x5c\ | ||||
| \xfc\x2e\x23\x13\xaf\x62\x8c\x2a\x6b\x77\xc8\xb1\x50\xf4\x01\x82\ | ||||
| \x65\x9d\xbd\xb4\x2c\xec\xfc\x70\x80\x70\x30\x41\xeb\xc2\x9e\xe2\ | ||||
| \x4a\xac\x2e\x22\x84\x40\x88\xe2\x5e\x56\x20\xd1\x81\x2d\xb8\x89\ | ||||
| \x6f\x63\x88\x31\x70\xae\x40\xdf\xa9\x3c\x52\xce\x57\xdc\x55\x1f\ | ||||
| \x8d\x69\xc9\x15\xce\x32\x70\xe1\x59\x26\x26\x7f\x39\x6b\xb3\x62\ | ||||
| \x58\x10\x8b\xd2\xbd\x64\x2d\x3b\xb7\xdc\x8b\x25\xad\x1a\x00\x42\ | ||||
| \x09\x16\xb7\xf6\x22\xa4\x44\xf8\xfd\x88\x59\x95\x1f\x80\x0a\xee\ | ||||
| \x42\x45\xbe\x04\x48\x8e\x7f\x90\xe7\xe2\xa8\x37\xb3\xd3\x9a\x7d\ | ||||
| \x0f\xa8\x65\x34\x0c\x30\x99\x3b\xc1\xe9\xa1\xbf\x40\xe9\xec\xcc\ | ||||
| \xf9\x68\x38\xce\xde\x7b\xf6\xf1\x8d\x87\x9f\x63\xfb\xc6\xcf\x57\ | ||||
| \xb5\x55\x06\x10\x74\x22\x74\xb6\xf6\x12\x0a\x26\x90\xaa\x1f\x74\ | ||||
| \xb2\x42\xf8\x02\x78\xd1\x47\xd1\xc1\x9d\x78\xae\xe6\xc0\x7b\x39\ | ||||
| \x92\x69\x55\x75\xbb\x58\xcb\x88\x18\xe0\x4a\xea\x75\x2e\x8d\xbd\ | ||||
| \x3c\x33\x0a\x52\x5a\x34\x35\x74\xd0\xb3\x64\x2d\xb1\x48\x5d\x6d\ | ||||
| \x00\x42\x48\x96\x76\xdc\x42\x5b\xd3\xcd\x08\x35\x84\xf4\x0e\x55\ | ||||
| \x74\xc9\xd8\xcb\xf1\x12\x4f\x61\x02\x6b\x19\x1b\xf7\x38\x74\x3c\ | ||||
| \x47\xbe\x30\x7f\xd1\x57\xa9\xad\xac\xec\x30\x1e\x67\x86\x7f\x48\ | ||||
| \x36\x7f\xaa\x06\xe4\x2a\x00\x00\x2d\x0d\x3d\xac\x58\xb2\x03\x5b\ | ||||
| \xe4\xb0\x0a\xaf\x82\xc9\x57\xbc\x50\x3b\xb7\xe2\xc5\x9e\x00\x21\ | ||||
| \x39\x3f\xec\x72\xf2\x74\x61\xa6\x34\xa8\x65\x03\x39\x7b\xdb\x2a\ | ||||
| \x80\xbc\x7b\x9e\xcb\xa9\xd7\x31\x15\xa6\x6e\xcd\x00\xb6\x1d\x60\ | ||||
| \xdb\xba\x07\x88\x86\xe3\x58\x85\xff\x40\xf8\xc7\xaa\x38\x24\x50\ | ||||
| \xa1\xdd\xa8\xc8\x83\x28\xa5\x39\xfa\x7e\x8e\xa1\x51\x0f\x4b\x56\ | ||||
| \xd2\xa4\xea\x9d\xbb\x54\x94\xce\x92\xca\x1c\xc6\x57\x93\xd7\x0f\ | ||||
| \x00\xd0\xd9\xda\xcb\x9d\x9b\xbe\x86\xe5\x1f\xc3\xc9\x3c\x87\x30\ | ||||
| \xa9\xca\x57\x8b\x70\x71\x2a\x85\x77\xe3\x7b\xb0\xff\x48\x96\xe4\ | ||||
| \x64\x79\x3e\x18\xe6\x26\x77\x29\x50\x39\x88\xc1\xf5\x46\xf1\x55\ | ||||
| \x95\xfe\x6a\x05\x00\xd8\xd2\xfb\x05\xda\x9a\x56\x60\xe5\x5f\xc4\ | ||||
| \x72\xdf\xa9\x3a\x2d\xb4\x68\xc3\x8b\xfe\x0e\x38\xdd\x24\x27\x7d\ | ||||
| \x8e\x9d\x2c\xe0\xba\xf3\xdf\x1f\xe6\x6e\x45\xab\x8f\xca\x75\x03\ | ||||
| \x2c\x5c\xd0\xc5\x9d\x1b\x7f\x9b\x70\xc0\xc3\x49\x3f\x85\xd0\x23\ | ||||
| \x95\x15\x85\x40\x05\xb7\xe1\xc5\x1e\xc7\x18\x8b\xd3\xe7\x0b\x9c\ | ||||
| \xb9\xe0\x5e\xc7\xf3\xa5\xa2\x38\x76\x3d\xb6\x15\xab\x59\xbf\x2a\ | ||||
| \x80\x65\x39\x6c\x59\x7b\x3f\x9b\xd7\xdc\x8f\xa3\x0e\x12\x48\x7d\ | ||||
| \x0b\xf4\x95\xb9\xfe\x03\x10\xc4\x8f\xfc\x26\x2a\xfa\x10\xbe\x27\ | ||||
| \x78\xaf\x3f\xc7\xc4\x3c\x4b\x6b\x69\xe4\x4b\xa7\x92\x63\xd5\xd1\ | ||||
| \x98\xb8\x03\xdb\xaa\xa3\x56\xa9\x0a\x20\x84\x20\x16\x69\xe4\xb3\ | ||||
| \xb7\x7d\x83\x25\x1d\xeb\xb1\xf2\xff\x84\x93\x7d\x0e\x4c\xa6\x0a\ | ||||
| \x84\x8d\x17\x7b\x0c\x13\xdc\x46\x26\xa3\x38\x74\x3c\x87\xef\x9b\ | ||||
| \xaa\xcb\x26\x15\xce\x37\x26\x6e\xa7\xb9\xfe\x33\x15\x8b\xb6\x6b\ | ||||
| \x06\x98\x96\xe6\x86\x65\xdc\xb3\xf3\x8f\x58\x10\x8d\x62\x4f\x7e\ | ||||
| \x1f\x3b\xf7\x42\xf5\x88\x5a\x8b\xf1\xe2\x8f\x21\x9c\x2e\x2e\x5c\ | ||||
| \x72\x39\x75\xce\x9d\xb7\xb8\x9b\x79\x44\x0f\x44\x83\xcb\x58\xd6\ | ||||
| \xfe\x38\x8e\xbd\xa0\x66\xe7\x6b\x02\x00\xf8\x44\xf7\x4e\xbe\x78\ | ||||
| \xd7\xf7\x68\x4c\x84\x71\x52\xfb\xb0\xb3\xcf\x03\xf9\x0a\x05\x9a\ | ||||
| \x8d\x0a\xfe\x1a\x5e\xec\xeb\x68\x1d\xa0\xef\x54\x9e\x89\x94\xfa\ | ||||
| \x90\x1b\x9b\xa4\x2e\xb2\x8e\x55\x4b\xbe\x47\x2c\xbc\xfc\x9a\x9c\ | ||||
| \xaf\x19\xc0\xb2\x1c\xd6\xaf\xfc\x2c\x77\xdf\xb6\x8f\x86\x58\x1c\ | ||||
| \x27\xf5\xfb\xd8\x99\xbf\x46\x98\xf2\x52\xa3\xe8\xa8\x83\x1f\xde\ | ||||
| \x8b\x76\x56\x93\x4c\x7b\x7c\x70\xa6\x80\x2e\x79\x4c\x72\x55\xaf\ | ||||
| \x28\x0d\x89\x1d\xac\xec\xfa\x36\xf5\xf1\x4d\x73\xf6\xbb\x1f\x19\ | ||||
| \x00\x40\xc0\x09\xb3\x7d\xfd\x17\xb9\xf7\xd3\x4f\x13\x0b\x7a\x38\ | ||||
| \xe9\xa7\x71\xd2\xcf\x80\xc9\xcd\xd1\x35\xb2\x11\x15\xf9\x0a\xc6\ | ||||
| \x08\x4e\x9f\x2f\xcc\x3c\xd5\x98\x2d\x8d\x89\x3b\x58\xd9\xf9\x14\ | ||||
| \x0b\x62\x1b\xae\xcb\xf9\x6b\x02\x00\x08\x06\xa2\x6c\x5e\x73\x1f\ | ||||
| \xbf\xf1\xeb\xdf\xa5\xa9\x2e\x4e\x20\xfb\x03\x02\xc9\x47\x11\xaa\ | ||||
| \x1f\xf0\xcb\x5e\x45\xa9\xe0\x1d\x08\xab\x89\x5c\x4e\x71\x7a\xd0\ | ||||
| \x2d\x5b\x91\x2c\x19\xa1\xbd\xf1\x01\x6e\xee\x7a\x96\x58\x78\xc5\ | ||||
| \x75\x39\x3e\x2d\xd7\xfc\xe8\xd8\xb6\x03\x6c\x5b\xbf\x97\xa6\x86\ | ||||
| \xa5\xbc\xfc\xf3\x3f\xe1\xe8\x07\xcf\x23\xbd\xc3\x78\x91\x47\x50\ | ||||
| \xe1\xbb\x40\xb6\x4e\x2d\x8f\x06\x84\x85\x90\x70\x79\xdc\x47\xa9\ | ||||
| \xe2\xbe\x22\xe8\xb4\xd3\xd5\xfa\x30\xed\x8d\xf7\x11\x70\x1a\x6e\ | ||||
| \xc8\xf9\xeb\x02\x28\x46\xd0\xe6\xe6\xa5\xb7\x51\x1f\x6f\xe3\xc5\ | ||||
| \x9f\xfd\x31\x87\xfa\x5e\x26\x97\x7a\x02\x55\x78\x0d\x15\xfe\x3c\ | ||||
| \xc6\xea\xc0\xce\xfd\x18\xa1\x47\x99\x79\x65\x22\x20\x12\xea\xa1\ | ||||
| \xbb\xfd\x71\x5a\x1a\x3e\x87\x14\x37\xf2\xd8\xfd\xaa\x88\x1b\xfd\ | ||||
| \xaf\x44\x6a\x72\x84\xc3\xfd\xaf\xf0\x93\xb7\xff\x9c\x8b\xa3\xfd\ | ||||
| \x28\x13\x06\x11\x06\x3d\x06\xc6\x43\x69\x58\xb3\xa2\x8e\x5d\x3b\ | ||||
| \xee\xa6\xab\xf9\x11\x12\xd1\x75\x48\x79\x6d\x2f\xb3\x3f\x56\x80\ | ||||
| \x69\x49\x67\x2e\xf3\xd3\x77\xfe\x8a\x5f\x1e\x7f\x91\x2b\xc9\x41\ | ||||
| \x72\x85\x14\x96\x74\xe8\x6c\x5b\xcc\x9e\x5f\xfd\x2d\x56\x76\x3d\ | ||||
| \x84\x65\x45\x6e\xbc\xa3\x8f\x0b\x00\x40\x29\x8f\xa1\xcb\x27\x39\ | ||||
| \x73\xe1\x20\x23\xe3\xa7\x89\x84\x63\xac\xee\xd9\x44\xfb\xc2\x4f\ | ||||
| \x22\x65\xe0\xc6\x3b\xf8\xb8\x01\xa6\xc5\x18\x83\x52\x2e\x42\x58\ | ||||
| \x58\xd6\x47\x33\xd7\xab\xc9\xff\x00\xbd\xd9\x0a\x65\xdf\x68\x9f\ | ||||
| \x0e\x00\x00\x00\x25\x74\x45\x58\x74\x64\x61\x74\x65\x3a\x63\x72\ | ||||
| \x65\x61\x74\x65\x00\x32\x30\x31\x34\x2d\x31\x31\x2d\x31\x30\x54\ | ||||
| \x30\x35\x3a\x32\x30\x3a\x30\x36\x2b\x30\x31\x3a\x30\x30\x8d\xd4\ | ||||
| \xe4\x1f\x00\x00\x00\x25\x74\x45\x58\x74\x64\x61\x74\x65\x3a\x6d\ | ||||
| \x6f\x64\x69\x66\x79\x00\x32\x30\x31\x34\x2d\x31\x30\x2d\x31\x36\ | ||||
| \x54\x30\x35\x3a\x32\x32\x3a\x31\x38\x2b\x30\x32\x3a\x30\x30\xf2\ | ||||
| \x2f\xb1\x0f\x00\x00\x00\x00\x49\x45\x4e\x44\xae\x42\x60\x82\ | ||||
| \x00\x00\x00\x06\x62\x4b\x47\x44\x00\xff\x00\xff\x00\xff\xa0\xbd\ | ||||
| \xa7\x93\x00\x00\x00\x09\x70\x48\x59\x73\x00\x00\x0b\x13\x00\x00\ | ||||
| \x0b\x13\x01\x00\x9a\x9c\x18\x00\x00\x00\x07\x74\x49\x4d\x45\x07\ | ||||
| \xdf\x04\x1b\x12\x2a\x1c\x39\xec\x95\x7d\x00\x00\x09\x5e\x49\x44\ | ||||
| \x41\x54\x68\xde\xd5\x9a\x7b\x8c\x54\xd5\x1d\xc7\x3f\xbf\xfb\x9a\ | ||||
| \x99\x65\x5f\xc8\x63\x41\x51\x40\x40\xac\xd1\x20\xd1\xc5\x68\x40\ | ||||
| \x51\xac\x6b\xb5\xe0\xa3\xb1\x94\xa8\x58\x6b\x2a\xad\xa6\x91\xc6\ | ||||
| \xd8\x54\xa5\x16\x9b\xf1\xd1\x2a\xa9\x68\x6b\x35\xd1\x1a\x27\xb6\ | ||||
| \x6b\xb5\xad\xb5\x69\x8b\x4b\x5c\xaa\x28\x0a\x2c\x2f\x95\x5d\x1e\ | ||||
| \x8b\x40\x61\xdd\x65\x60\xd8\xf7\xee\x9d\xb9\xaf\xd3\x3f\x76\x06\ | ||||
| \x76\x61\x76\x65\x67\xc6\x3f\xfc\x25\x37\x33\x39\xf7\xce\xf9\x9d\ | ||||
| \xef\xf9\x9e\xdf\x73\xae\x28\xa5\xf8\x3a\x8b\x51\xc8\xc9\x6a\xea\ | ||||
| \x2a\x8a\x95\xa2\xa8\xb4\x54\xc2\xe5\x65\x9a\x15\xb6\x24\xec\x78\ | ||||
| \x4a\xab\x6f\x70\xb5\xc6\x3d\x2e\x4d\xcd\x01\x89\xa3\x81\x4a\x24\ | ||||
| \x7c\x3f\xbd\x6f\x3d\x40\x2b\xd0\x59\x5b\x6d\xe7\xb4\x93\x32\x1c\ | ||||
| \x06\x6a\xea\x2a\x42\xc0\x4c\xe0\x7c\xe0\x1c\xe0\x6c\xe0\x2c\x60\ | ||||
| \x14\x10\x12\x41\xb7\x2c\xd1\x5a\x0e\xf9\xda\x8e\x9d\xae\x1c\x68\ | ||||
| \xf2\xb4\x44\x22\x10\xcf\x03\xa5\x14\x41\x00\x4a\x81\x52\x64\x94\ | ||||
| \x06\x80\x97\xbe\x6c\xe0\x20\x50\x0f\x6c\x04\x56\xd7\x56\xdb\xf1\ | ||||
| \x9c\x01\xd4\xd4\x55\x14\xa7\x17\x76\x05\x70\x3d\x70\x25\x30\x66\ | ||||
| \xa8\xc9\xbe\x68\xf6\x59\xfd\x6e\x92\x2f\x9a\x5d\x74\x5d\x10\xcd\ | ||||
| \x02\x31\x40\x04\x85\x96\x7e\x4a\x21\x4a\x01\x01\x22\x0a\x51\x6e\ | ||||
| \x7a\xfd\x00\x72\xc2\x27\xfb\x81\x95\xc0\x9b\x40\x4b\x6d\xb5\x1d\ | ||||
| \x7c\x29\x80\x9a\xba\x8a\x31\xc0\xf7\x81\xf9\xc0\xec\xfe\xb3\x0d\ | ||||
| \xba\x0b\x02\x07\x9b\x7c\x5e\x7f\xa3\x87\xc0\x4b\x62\x5b\x33\xe8\ | ||||
| \xd4\x66\x91\xd4\x26\xe3\x4a\x39\x81\x84\x51\x98\x7d\x53\x29\x1f\ | ||||
| \x4d\x7c\xcc\xe2\x22\x4c\xd3\x47\xf7\x5b\x31\xbc\x38\xa6\xdb\x84\ | ||||
| \xe5\xee\x23\xe4\xec\xc2\x70\x0f\xa1\x44\x50\x62\x65\xd4\xb7\x01\ | ||||
| \x2f\x03\xbf\xae\xad\xb6\x13\x83\x02\xa8\xa9\xab\xb8\x00\xf8\x37\ | ||||
| \x70\xe6\x70\xce\xa1\xae\xc3\xaa\xd5\x49\x36\x6f\x4e\xd2\x3c\xe2\ | ||||
| \x61\xda\x8c\x2b\xf0\x89\x00\x1a\x30\x70\x83\xc4\xb0\x30\x47\x4f\ | ||||
| \x40\x8c\xd0\x80\xbd\x11\xe5\x20\x2a\x89\xa6\x6c\x42\xce\x5e\xca\ | ||||
| \x3b\xfe\x42\x79\x47\x0c\x94\x8f\x12\x33\xf3\xd8\x27\xc0\xdc\xda\ | ||||
| \x6a\xbb\x3d\x33\x90\xe1\x95\x35\x5b\xc7\x45\x80\x2d\xc3\x5d\x3c\ | ||||
| \xf4\x9d\xeb\x33\x4e\x17\x12\xc6\x0d\x24\x8c\xaa\xf4\xe2\x33\x47\ | ||||
| \x5c\x1d\xbb\xc4\x0c\x61\x8d\x9d\x88\x18\x56\x7a\x2c\x38\x76\x29\ | ||||
| \x31\x08\xb4\x62\x3c\x7d\x0c\x3d\x91\x4b\x69\x1a\xff\x7b\x1a\xce\ | ||||
| \x69\xa1\xbd\xfc\x8e\x34\x13\x0a\x60\x06\xf0\x5e\x7f\xdd\xc7\x00\ | ||||
| \xb4\x75\xf8\x4b\x72\xf5\x4a\x41\x00\xe7\x4e\x33\x99\x32\x7d\x2c\ | ||||
| \x0a\x3d\x2b\x42\x2d\x54\x84\x35\x76\x22\xe8\xa7\xa2\x22\x40\x54\ | ||||
| \x0a\x25\x16\x4d\xe3\x5f\xa4\x65\xdc\x4a\xe4\x38\x9b\x33\xe6\x2d\ | ||||
| \x8a\xac\x38\x09\x40\xed\x47\x5d\xd3\x74\x4d\x72\x76\xa1\xba\xa1\ | ||||
| \x71\xc7\xd5\xef\x30\xa6\xf8\xf0\xc9\x36\x62\x58\x18\xa7\x9d\x0e\ | ||||
| \xa2\xf5\xd1\x75\xea\xdc\x22\xca\xa6\xad\x6c\x21\x2d\xe3\x56\xa0\ | ||||
| \x05\xa9\xcc\x8d\x5b\xe7\x2d\x8a\x9c\x3e\x90\x81\x4e\xdf\xde\x52\ | ||||
| \xdf\x8b\xa6\xe5\x1e\x07\x4a\x23\xed\x3c\x32\xff\x21\x4c\xdd\x43\ | ||||
| \xa9\xe3\x9b\xa1\x17\x9f\x86\xe8\x66\xce\xf3\x8a\xf2\xe8\x28\x9e\ | ||||
| \x8f\x63\x4d\xcb\xb0\x50\x0e\x5c\x3a\x00\x80\x69\xc8\xaa\x86\xc6\ | ||||
| \x24\x4d\x2d\x2e\x92\x23\x11\x7e\xa0\x33\x7e\xe4\x01\x1e\xb8\x36\ | ||||
| \x8a\x65\xb8\x28\x04\xd1\x74\xb4\x48\x49\xde\x41\x32\xd0\xc2\xd8\ | ||||
| \x91\x0b\x33\x00\x42\xc0\x94\x01\x00\x62\x51\xa7\x36\x50\x34\xac\ | ||||
| \xdf\xd6\x8b\x9d\x0a\x72\x56\xe4\xf9\x26\x17\x4c\xd8\xcc\xc2\x59\ | ||||
| \xaf\xa1\x49\x00\x22\x88\x61\x92\xbf\x18\xf8\xfa\x69\xfd\x07\xca\ | ||||
| \x07\x00\x48\xcb\x2d\x3d\xbd\x81\xbf\xe6\xa3\xee\xbc\xd5\xdd\x34\ | ||||
| \xf3\x75\x2e\x99\xbc\x1e\x3f\x18\xee\xb9\x1f\xea\x28\xf9\x27\x8d\ | ||||
| \x9d\x08\x60\x87\xa6\xf1\x58\xa2\xd5\xe3\x83\x4d\x3d\x27\xba\xf0\ | ||||
| \x61\x89\xeb\x1b\x2c\xfd\xe6\xe3\xcc\x9a\xb4\x9e\xa4\x5d\x80\xc5\ | ||||
| \xe3\xa2\xfb\x03\x62\x58\xc7\x49\x00\x62\x51\x47\x01\x51\x5d\x97\ | ||||
| \x86\x03\x5f\x38\x34\xec\x49\xe6\xac\x50\x21\x80\x62\xc9\xe5\x2b\ | ||||
| \x38\x6f\xd4\x7a\x02\xa5\xe5\xb9\xfb\x0e\xa6\xd7\x92\x09\x7e\x2e\ | ||||
| \xd0\x9c\x8d\x01\x62\x51\xc7\x05\xaa\x00\x7f\x6b\x83\xcd\x91\x56\ | ||||
| \x2f\x67\xa3\x56\x08\x25\x91\x76\xee\x9e\xb5\x9c\xd1\x91\x96\x34\ | ||||
| \xa8\x5c\x01\xb8\xe8\xde\x31\x17\x6d\x03\x7b\xb2\x02\x48\x83\x68\ | ||||
| \x02\x16\x2a\x05\xff\x5d\xdf\x4d\x47\x97\x9f\xb3\x7b\x55\x4a\xe3\ | ||||
| \xcc\xb2\x06\xee\xbb\xf8\xc7\x94\x85\x5a\xf1\x03\x23\x67\x00\xa6\ | ||||
| \xd7\x9c\x61\x20\x05\xec\x1e\x14\x40\x5a\xfe\x21\xc2\x8b\x8e\xa3\ | ||||
| \x78\x7f\x43\x37\xc9\x94\xca\xc3\x1e\x42\x4c\x2e\xdf\xc6\xcf\x2f\ | ||||
| \xb9\x8d\xb2\x70\x6b\x0e\x4c\x08\x96\xd7\x82\x1c\x0f\x64\x89\xda\ | ||||
| \x6a\xfb\xe8\x90\x00\x62\x51\xc7\x07\x1e\x16\x61\x5f\x47\x57\xc0\ | ||||
| \xe6\xed\xbd\x04\xb9\x7b\x57\x02\xa5\x31\xa1\x74\x07\x3f\xbd\xf8\ | ||||
| \x6e\x46\x47\x5a\x00\x49\xa6\xcf\xf2\x97\xb3\x28\x3a\xa1\x64\x7d\ | ||||
| \xff\xa1\x0f\x07\xf3\x42\x27\x82\x38\x7a\xb4\xdd\xbd\x51\xd7\x84\ | ||||
| \x3d\xfb\x1d\x76\xed\x4b\x62\x18\x92\x17\x88\x73\x47\xad\xe3\xa1\ | ||||
| \x4b\x17\x71\xcb\xb9\x4f\xbd\xdb\x91\x2a\xb9\x0d\xb8\x1f\xf8\x6d\ | ||||
| \x3a\x0b\x3e\x38\x48\xa2\x42\x24\xf5\x29\xea\xb8\x31\xae\x39\xa5\ | ||||
| \x8a\xac\xe1\xc0\xb7\xaf\xd7\x34\x79\xb4\x7e\xcf\xa1\x8b\xfe\xba\ | ||||
| \x6a\x0b\x86\xa1\x33\xf7\x92\x62\x26\x4e\xb0\xf0\x7d\x95\x87\x4b\ | ||||
| \x54\x20\xaa\x47\x17\xff\xee\x79\x17\xb5\xfe\x79\xfa\x13\xfb\xa5\ | ||||
| \x33\x35\x2a\x0c\x8c\x00\xc6\x01\xdf\x03\x7e\x94\x2e\xa8\x08\xb4\ | ||||
| \x22\xa6\xed\xbb\x9c\xb0\xbd\x11\x25\x06\xc0\xa8\xda\x6a\xbb\xf5\ | ||||
| \x24\x00\xbf\x7b\x63\xba\x5c\x36\x73\x52\x79\x24\x6c\xcd\x57\x4a\ | ||||
| \x2d\x07\x26\x83\x22\x50\x8a\xb7\x6b\xf7\xb0\xf3\xf3\xfd\x58\xa6\ | ||||
| \xf0\xad\x2b\x4a\x28\x29\xd6\x0b\x11\x9b\x5c\xe0\xca\xaa\xca\xf8\ | ||||
| \xba\x6c\x37\xc7\x2f\xef\xba\x13\x58\x11\x68\xc5\x23\x2f\xd8\x59\ | ||||
| \x8a\x52\x2e\x20\x87\x6b\xab\xed\x8a\xac\x47\x68\xe7\xde\x43\xa5\ | ||||
| \x1f\x6c\xfa\x7c\x75\x7b\x67\xef\xab\x96\xa9\x4f\xb6\x0c\x9b\xce\ | ||||
| \xe4\x18\x3e\xde\xfb\x5d\x5a\xb4\x18\x18\x67\xe1\xb8\x01\x1f\x6f\ | ||||
| \xeb\xcd\x8b\x81\x7e\x62\x02\xab\x6b\xea\x2a\xae\xca\x76\xb3\x65\ | ||||
| \x79\xc9\x2b\x81\x16\x59\x10\x72\x0e\xf4\x88\xdf\x9d\xf1\x40\xff\ | ||||
| \x19\xb4\x22\xbb\xfd\x61\xb3\x5c\x44\xde\x0b\x59\xc6\x8c\xf3\xa7\ | ||||
| \x4f\x45\x95\x3f\xc6\xb6\x83\x73\xe8\x49\x95\x11\x20\xe8\xee\x27\ | ||||
| \x44\x8e\x5c\x88\xaf\x4c\xa6\x4d\xb2\x98\x53\x59\x8c\xeb\x15\x04\ | ||||
| \xc8\x61\xe0\xda\xaa\xca\xf8\xd6\x6c\x37\xcf\xbf\xef\xb9\xbf\x8d\ | ||||
| \x8b\x2f\xbd\xb9\xaf\xb0\xe1\xe6\xda\x6a\xfb\xad\xac\x0c\x88\x48\ | ||||
| \x0a\x38\x90\x72\x3c\x36\x6e\xef\x64\x6d\xe3\x7c\xba\x53\xe5\x7d\ | ||||
| \x59\x25\x10\x58\x33\x48\x95\xbf\x80\x26\x2e\xbb\xf6\xa6\x68\xd8\ | ||||
| \x93\xcc\x2b\xfd\xee\x27\x63\x81\x97\x6b\xea\x2a\x46\x67\xbd\x79\ | ||||
| \xe4\x91\x86\xf4\xe2\x03\x60\xfd\x50\xb9\x50\xf2\x58\x84\x13\x1f\ | ||||
| \xbd\x2f\xdd\xe8\x1f\x5a\xf1\x8a\x7e\x88\x1f\x59\x88\x61\xc0\xe6\ | ||||
| \xcf\x6c\x0e\x1d\xc9\x3d\x52\x9f\x20\x33\x81\x97\xb2\x07\x31\x7b\ | ||||
| \x6a\xfa\xeb\x26\xa0\x6b\x50\x00\xe9\x5c\x68\x7b\x5f\x9f\x23\x85\ | ||||
| \xf8\x4d\x59\xa6\xd3\x70\x4a\x9f\x41\x69\x63\xf0\x03\xc5\x86\x6d\ | ||||
| \xbd\x85\x3a\x46\x00\x37\xd4\xd4\x55\x3c\x93\x05\xc2\xbd\xc0\x54\ | ||||
| \xe0\xa6\x74\x33\x6c\xf0\xb6\xca\xe2\x65\xd6\x37\x80\x75\x48\x68\ | ||||
| \xa4\x53\xf6\x2a\x5e\xe4\xe6\x6c\x7e\x10\x3d\xb9\x86\x50\xeb\x35\ | ||||
| \xf8\xbe\x70\xf6\x59\x21\xe6\x54\x8e\x28\x54\xd6\x6c\xa7\xed\x61\ | ||||
| \xed\xa9\x3c\x9c\x2d\x99\xdb\x01\x34\xa2\x92\x68\xee\xa6\xc1\xb2\ | ||||
| \x34\xbc\xf0\x55\xb8\x25\x4f\xa1\x6b\x1e\x9f\xff\x2f\xc5\xa7\x3b\ | ||||
| \x93\xe8\x05\xf1\xac\x44\x80\xfb\xd3\x5d\xc0\xe1\x03\x48\xcb\xd3\ | ||||
| \x00\xba\x53\x8b\xa8\xce\x41\x92\x2b\x70\x47\xdc\x83\x1f\xb9\x05\ | ||||
| \xc3\x50\x34\x34\x26\x39\xd8\xec\x16\xca\x1e\x16\x00\x13\x73\x06\ | ||||
| \x10\x8b\x3a\x6f\x82\x34\x6b\xce\x46\x34\xf7\xb3\x21\x42\x6a\x08\ | ||||
| \xb7\xf4\x09\x94\x76\x06\xae\x17\xb0\xb5\xde\xc6\x4e\x16\xcc\x1e\ | ||||
| \xee\xcc\x87\x01\x80\x9f\x28\x11\xac\x8e\x7b\x87\x6c\x2e\x06\xc6\ | ||||
| \x14\x9c\xb2\x17\x11\xf1\x48\xb4\x7b\x6c\xfa\xac\x17\x5d\x2f\x08\ | ||||
| \x0d\xf3\xf3\x05\xb0\x1a\x8c\x0f\xc4\xfd\x04\xb3\xfb\xe9\xc1\x41\ | ||||
| \x28\xf0\xc3\xd7\xe2\x96\x3e\x8f\xa1\x79\xec\x6b\x72\xd8\xd6\xd0\ | ||||
| \x8b\x69\xe4\x0d\x62\x52\x5e\x00\x62\x51\xa7\x1b\x78\x06\x31\x1d\ | ||||
| \xa3\xeb\x49\x34\x77\xf7\xd0\x49\x4d\xd1\x0f\xf0\xc3\x37\xa2\x89\ | ||||
| \xa2\xa1\x31\x45\xcb\x11\x37\x5f\x00\xad\xf9\x32\x40\x2c\xea\xfc\ | ||||
| \x1d\x78\x5b\x54\x07\x66\xe7\x03\x80\x3f\xb4\x3d\x94\x3d\x85\xd2\ | ||||
| \x46\xe3\x7a\x8a\x2d\xdb\xed\xbc\xea\x07\x60\x55\xde\x00\x32\x6d\ | ||||
| \x3c\x90\x16\x3d\xf9\x4f\xac\xae\x28\x6a\x48\x7b\x98\x8a\x53\xfe\ | ||||
| \x1a\x9a\x72\x89\x27\x3c\xea\x1b\x93\xf9\xd8\xc3\xca\x82\x00\x48\ | ||||
| \x17\xf9\xf3\x10\x33\x69\x76\x2d\xc7\xec\xad\x1e\xaa\x8a\xc7\x0f\ | ||||
| \xcf\xc3\x2d\x7d\x14\x4d\x5c\x36\x6f\xef\x6b\x0a\xe4\xd0\x72\x7d\ | ||||
| \x21\x53\xf3\x16\x82\x81\x4c\x70\xbb\x57\x89\x89\xd5\x71\x17\x46\ | ||||
| \xf2\xed\x21\x41\x38\x25\x0f\x12\x84\xae\xc1\xd4\x03\xd6\x6d\xea\ | ||||
| \x26\xe5\x0e\xcb\xb5\x6e\x04\x7e\x51\x55\x19\xf7\x0a\x06\x20\x0d\ | ||||
| \xe2\x8f\xc0\xaf\x50\x2e\x56\xdb\xed\xe8\xce\x87\x43\xb8\x57\x13\ | ||||
| \xa7\xf4\x71\xd0\x4a\xe9\xec\x0e\xd8\xb5\x37\x75\xaa\x6a\xb6\x02\ | ||||
| \x57\x55\x55\xc6\x13\xa7\xfa\x83\x61\x25\xc3\xb1\xa8\xf3\x4b\x90\ | ||||
| \x95\x60\x13\x4e\xcc\x41\x4f\xbe\x33\x78\xa3\xd7\xba\x08\x2f\xbc\ | ||||
| \x10\x94\xcb\xee\x7d\x29\xec\xe4\x97\x5a\xf4\xfb\xc0\x65\x55\x95\ | ||||
| \xf1\x9e\xe1\xac\x69\xd8\xd9\x7c\x2c\xea\x2c\x05\x79\x52\x89\x49\ | ||||
| \xa8\x75\x01\x66\xf7\x8a\xc1\x53\x8d\xb2\xdf\x80\x68\x74\x75\xfb\ | ||||
| \xec\xdc\x9b\x62\x88\xff\x1f\x5e\x06\x16\x54\x55\xc6\x87\xdd\x0a\ | ||||
| \xcc\xa9\x1c\x89\x45\x9d\x07\x81\xfb\x00\xcc\xae\x07\x09\xb7\x7d\ | ||||
| \x27\xbb\x39\x68\x25\x04\xd6\xd5\x18\xba\x47\xfd\xee\x24\x9e\x9f\ | ||||
| \x95\x85\x7b\x80\x25\x55\x95\xf1\xce\x5c\xd6\x92\x73\x3d\x15\x8b\ | ||||
| \x3a\xcf\xa6\xc3\x7d\x97\x96\x7c\x8b\x48\x7c\x22\xba\xbb\x81\xe3\ | ||||
| \x7f\x99\x66\x40\x8c\xea\x3b\x52\x81\xe2\xf0\xd1\x01\x1d\xbe\x36\ | ||||
| \xe0\xb6\xaa\xca\xf8\x1f\xaa\x2a\xe3\x7e\xae\xeb\xc8\xab\x20\x8c\ | ||||
| \x45\x9d\x7f\x01\x73\xc1\x58\x2b\x41\x13\xa1\xc4\x5c\x42\xed\x4b\ | ||||
| \xd0\x53\x35\x48\x70\x18\x3d\xb5\x16\x3d\xb5\x0a\xc4\x40\xa0\x7f\ | ||||
| \x23\x60\x03\x70\x5d\x55\x65\xfc\x4f\xf9\xb7\xdc\x0b\x50\x85\x2c\ | ||||
| \x5e\x66\x15\x01\x77\x01\xcf\x42\x00\x52\x8c\x92\x11\x88\xea\x01\ | ||||
| \xd5\xd3\xd7\x1c\xf7\x14\xb7\xde\x30\x12\xcb\x94\x97\x94\xe2\x67\ | ||||
| \x55\x95\xf1\xb6\x42\x64\x7c\x52\xc8\x97\x3d\x16\x2f\xb3\xa6\x00\ | ||||
| \xaf\xa4\xeb\xdb\xe2\xe3\x0d\x5e\x38\x67\x72\xa8\x73\x4e\xe5\x88\ | ||||
| \x65\x73\x67\x1c\x7a\x8e\x02\x8a\x7c\x15\x6f\xab\x2c\x5e\x66\xcd\ | ||||
| \x06\xae\x03\x2e\x04\xcc\xd2\x62\x6d\xfb\x35\xb3\x4b\x9f\x5f\x30\ | ||||
| \xfb\x70\x63\xa1\x75\xc9\x57\xf9\xba\xcd\xe2\x65\x56\x91\x08\x32\ | ||||
| \xf3\xbc\xb0\xbd\x74\x51\x67\xf0\x55\xe8\xf8\x3f\x66\x47\x07\xd4\ | ||||
| \xf8\xcc\x73\x31\x00\x00\x00\x00\x49\x45\x4e\x44\xae\x42\x60\x82\ | ||||
| \ | ||||
| " | ||||
|  | ||||
| qt_resource_name = "\ | ||||
| qt_resource_name = b"\ | ||||
| \x00\x06\ | ||||
| \x07\x03\x7d\xc3\ | ||||
| \x00\x69\ | ||||
| @@ -235,7 +187,7 @@ qt_resource_name = "\ | ||||
| \x00\x64\x00\x73\x00\x2e\x00\x70\x00\x6e\x00\x67\ | ||||
| " | ||||
|  | ||||
| qt_resource_struct = "\ | ||||
| qt_resource_struct = b"\ | ||||
| \x00\x00\x00\x00\x00\x02\x00\x00\x00\x01\x00\x00\x00\x01\ | ||||
| \x00\x00\x00\x00\x00\x02\x00\x00\x00\x01\x00\x00\x00\x02\ | ||||
| \x00\x00\x00\x12\x00\x02\x00\x00\x00\x01\x00\x00\x00\x03\ | ||||
|   | ||||
| @@ -88,7 +88,7 @@ | ||||
|    <item> | ||||
|     <widget class="QTabWidget" name="tabWidget"> | ||||
|      <property name="currentIndex"> | ||||
|       <number>0</number> | ||||
|       <number>2</number> | ||||
|      </property> | ||||
|      <widget class="QWidget" name="aboutTab"> | ||||
|       <attribute name="title"> | ||||
| @@ -109,7 +109,7 @@ | ||||
| p, li { white-space: pre-wrap; } | ||||
| </style></head><body style=" font-family:'Verdana'; font-size:9pt; font-weight:400; font-style:normal;"> | ||||
| <p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:'Sans Serif';"><br /></p> | ||||
| <p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Sans Serif'; font-weight:600;">(c) 2014, Virtual Cable S.L.U.</span></p> | ||||
| <p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Sans Serif'; font-weight:600;">(c) 2012-2016, Virtual Cable S.L.U.</span></p> | ||||
| <p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:'Sans Serif'; font-style:italic;"><br /></p> | ||||
| <p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><a href="http://www.udsenterprise.com"><span style=" font-family:'MS Shell Dlg 2'; font-size:8pt; text-decoration: underline; color:#0000ff;">http://www.udsenterprise.com</span></a></p> | ||||
| <p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><a href="http://www.openuds.org"><span style=" font-family:'MS Shell Dlg 2'; font-size:8pt; text-decoration: underline; color:#0000ff;">http://www.openuds.org</span></a></p> | ||||
| @@ -167,7 +167,7 @@ p, li { white-space: pre-wrap; } | ||||
| <html><head><meta name="qrichtext" content="1" /><style type="text/css"> | ||||
| p, li { white-space: pre-wrap; } | ||||
| </style></head><body style=" font-family:'Verdana'; font-size:9pt; font-weight:400; font-style:normal;"> | ||||
| <p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'MS Shell Dlg 2'; font-size:8pt;">Copyright (c) 2014 Virtual Cable S.L.</span></p> | ||||
| <p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'MS Shell Dlg 2'; font-size:8pt;">Copyright (c) 2012-2016 Virtual Cable S.L.</span></p> | ||||
| <p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:'MS Shell Dlg 2'; font-size:8pt;"><br /></p> | ||||
| <p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'MS Shell Dlg 2'; font-size:8pt;">All rights reserved.</span></p> | ||||
| <p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:'MS Shell Dlg 2'; font-size:8pt;"><br /></p> | ||||
|   | ||||
| @@ -1,154 +1,164 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # Form implementation generated from reading ui file 'about-dialog.ui' | ||||
| # | ||||
| # Created: Tue Dec 09 10:54:30 2014 | ||||
| #      by: PyQt4 UI code generator 4.11.2 | ||||
| # | ||||
| # WARNING! All changes made in this file will be lost! | ||||
|  | ||||
| from PyQt4 import QtCore, QtGui | ||||
|  | ||||
| try: | ||||
|     _fromUtf8 = QtCore.QString.fromUtf8 | ||||
| except AttributeError: | ||||
|     def _fromUtf8(s): | ||||
|         return s | ||||
|  | ||||
| try: | ||||
|     _encoding = QtGui.QApplication.UnicodeUTF8 | ||||
|     def _translate(context, text, disambig): | ||||
|         return QtGui.QApplication.translate(context, text, disambig, _encoding) | ||||
| except AttributeError: | ||||
|     def _translate(context, text, disambig): | ||||
|         return QtGui.QApplication.translate(context, text, disambig) | ||||
|  | ||||
| class Ui_UDSAboutDialog(object): | ||||
|     def setupUi(self, UDSAboutDialog): | ||||
|         UDSAboutDialog.setObjectName(_fromUtf8("UDSAboutDialog")) | ||||
|         UDSAboutDialog.resize(466, 402) | ||||
|         font = QtGui.QFont() | ||||
|         font.setFamily(_fromUtf8("Verdana")) | ||||
|         font.setPointSize(9) | ||||
|         UDSAboutDialog.setFont(font) | ||||
|         UDSAboutDialog.setLocale(QtCore.QLocale(QtCore.QLocale.English, QtCore.QLocale.UnitedStates)) | ||||
|         UDSAboutDialog.setModal(True) | ||||
|         self.vboxlayout = QtGui.QVBoxLayout(UDSAboutDialog) | ||||
|         self.vboxlayout.setSpacing(9) | ||||
|         self.vboxlayout.setMargin(9) | ||||
|         self.vboxlayout.setObjectName(_fromUtf8("vboxlayout")) | ||||
|         self.LogoLabel = QtGui.QLabel(UDSAboutDialog) | ||||
|         self.LogoLabel.setObjectName(_fromUtf8("LogoLabel")) | ||||
|         self.vboxlayout.addWidget(self.LogoLabel) | ||||
|         spacerItem = QtGui.QSpacerItem(20, 5, QtGui.QSizePolicy.Minimum, QtGui.QSizePolicy.Fixed) | ||||
|         self.vboxlayout.addItem(spacerItem) | ||||
|         self.TitleLabel = QtGui.QLabel(UDSAboutDialog) | ||||
|         self.TitleLabel.setObjectName(_fromUtf8("TitleLabel")) | ||||
|         self.vboxlayout.addWidget(self.TitleLabel) | ||||
|         self.VersionLabel = QtGui.QLabel(UDSAboutDialog) | ||||
|         self.VersionLabel.setObjectName(_fromUtf8("VersionLabel")) | ||||
|         self.vboxlayout.addWidget(self.VersionLabel) | ||||
|         spacerItem1 = QtGui.QSpacerItem(20, 5, QtGui.QSizePolicy.Minimum, QtGui.QSizePolicy.Fixed) | ||||
|         self.vboxlayout.addItem(spacerItem1) | ||||
|         self.tabWidget = QtGui.QTabWidget(UDSAboutDialog) | ||||
|         self.tabWidget.setObjectName(_fromUtf8("tabWidget")) | ||||
|         self.aboutTab = QtGui.QWidget() | ||||
|         self.aboutTab.setObjectName(_fromUtf8("aboutTab")) | ||||
|         self.vboxlayout1 = QtGui.QVBoxLayout(self.aboutTab) | ||||
|         self.vboxlayout1.setSpacing(6) | ||||
|         self.vboxlayout1.setMargin(9) | ||||
|         self.vboxlayout1.setObjectName(_fromUtf8("vboxlayout1")) | ||||
|         self.aboutBrowser = QtGui.QTextBrowser(self.aboutTab) | ||||
|         self.aboutBrowser.setOpenExternalLinks(True) | ||||
|         self.aboutBrowser.setObjectName(_fromUtf8("aboutBrowser")) | ||||
|         self.vboxlayout1.addWidget(self.aboutBrowser) | ||||
|         self.tabWidget.addTab(self.aboutTab, _fromUtf8("")) | ||||
|         self.authorsTab = QtGui.QWidget() | ||||
|         self.authorsTab.setObjectName(_fromUtf8("authorsTab")) | ||||
|         self.vboxlayout2 = QtGui.QVBoxLayout(self.authorsTab) | ||||
|         self.vboxlayout2.setSpacing(6) | ||||
|         self.vboxlayout2.setMargin(9) | ||||
|         self.vboxlayout2.setObjectName(_fromUtf8("vboxlayout2")) | ||||
|         self.authorsBrowser = QtGui.QTextBrowser(self.authorsTab) | ||||
|         self.authorsBrowser.setOpenExternalLinks(True) | ||||
|         self.authorsBrowser.setObjectName(_fromUtf8("authorsBrowser")) | ||||
|         self.vboxlayout2.addWidget(self.authorsBrowser) | ||||
|         self.tabWidget.addTab(self.authorsTab, _fromUtf8("")) | ||||
|         self.licenseTab = QtGui.QWidget() | ||||
|         self.licenseTab.setObjectName(_fromUtf8("licenseTab")) | ||||
|         self.vboxlayout3 = QtGui.QVBoxLayout(self.licenseTab) | ||||
|         self.vboxlayout3.setSpacing(6) | ||||
|         self.vboxlayout3.setMargin(9) | ||||
|         self.vboxlayout3.setObjectName(_fromUtf8("vboxlayout3")) | ||||
|         self.licenseBrowser = QtGui.QTextBrowser(self.licenseTab) | ||||
|         self.licenseBrowser.setObjectName(_fromUtf8("licenseBrowser")) | ||||
|         self.vboxlayout3.addWidget(self.licenseBrowser) | ||||
|         self.tabWidget.addTab(self.licenseTab, _fromUtf8("")) | ||||
|         self.vboxlayout.addWidget(self.tabWidget) | ||||
|         self.buttonBox = QtGui.QDialogButtonBox(UDSAboutDialog) | ||||
|         self.buttonBox.setOrientation(QtCore.Qt.Horizontal) | ||||
|         self.buttonBox.setStandardButtons(QtGui.QDialogButtonBox.Close) | ||||
|         self.buttonBox.setObjectName(_fromUtf8("buttonBox")) | ||||
|         self.vboxlayout.addWidget(self.buttonBox) | ||||
|  | ||||
|         self.retranslateUi(UDSAboutDialog) | ||||
|         self.tabWidget.setCurrentIndex(0) | ||||
|         QtCore.QObject.connect(self.buttonBox, QtCore.SIGNAL(_fromUtf8("clicked(QAbstractButton*)")), UDSAboutDialog.closeDialog) | ||||
|         QtCore.QMetaObject.connectSlotsByName(UDSAboutDialog) | ||||
|  | ||||
|     def retranslateUi(self, UDSAboutDialog): | ||||
|         UDSAboutDialog.setWindowTitle(_translate("UDSAboutDialog", "About UDS Actor", None)) | ||||
|         self.LogoLabel.setText(_translate("UDSAboutDialog", "<html><head/><body><p><img src=\":/images/img/uds.png\"/> UDS Actor Tools</p></body></html>", None)) | ||||
|         self.TitleLabel.setText(_translate("UDSAboutDialog", "<html><head/><body><p><span style=\" font-family:\'Sans Serif\'; font-size:9pt; font-weight:600;\">UDS Actor</span></p></body></html>", None)) | ||||
|         self.VersionLabel.setText(_translate("UDSAboutDialog", "Version 1.7.0", None)) | ||||
|         self.aboutBrowser.setHtml(_translate("UDSAboutDialog", "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0//EN\" \"http://www.w3.org/TR/REC-html40/strict.dtd\">\n" | ||||
| "<html><head><meta name=\"qrichtext\" content=\"1\" /><style type=\"text/css\">\n" | ||||
| "p, li { white-space: pre-wrap; }\n" | ||||
| "</style></head><body style=\" font-family:\'Verdana\'; font-size:9pt; font-weight:400; font-style:normal;\">\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'Sans Serif\';\"><br /></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'Sans Serif\'; font-weight:600;\">(c) 2014, Virtual Cable S.L.U.</span></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'Sans Serif\'; font-style:italic;\"><br /></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><a href=\"http://www.udsenterprise.com\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt; text-decoration: underline; color:#0000ff;\">http://www.udsenterprise.com</span></a></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><a href=\"http://www.openuds.org\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt; text-decoration: underline; color:#0000ff;\">http://www.openuds.org</span></a></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'Sans Serif\';\"><br /></p></body></html>", None)) | ||||
|         self.tabWidget.setTabText(self.tabWidget.indexOf(self.aboutTab), _translate("UDSAboutDialog", "&About", None)) | ||||
|         self.authorsBrowser.setHtml(_translate("UDSAboutDialog", "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0//EN\" \"http://www.w3.org/TR/REC-html40/strict.dtd\">\n" | ||||
| "<html><head><meta name=\"qrichtext\" content=\"1\" /><style type=\"text/css\">\n" | ||||
| "p, li { white-space: pre-wrap; }\n" | ||||
| "</style></head><body style=\" font-family:\'Verdana\'; font-size:9pt; font-weight:400; font-style:normal;\">\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'Sans Serif\';\">Adolfo Gómez García <agomez@virtualcable.es></span></p></body></html>", None)) | ||||
|         self.tabWidget.setTabText(self.tabWidget.indexOf(self.authorsTab), _translate("UDSAboutDialog", "A&uthors", None)) | ||||
|         self.licenseBrowser.setHtml(_translate("UDSAboutDialog", "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0//EN\" \"http://www.w3.org/TR/REC-html40/strict.dtd\">\n" | ||||
| "<html><head><meta name=\"qrichtext\" content=\"1\" /><style type=\"text/css\">\n" | ||||
| "p, li { white-space: pre-wrap; }\n" | ||||
| "</style></head><body style=\" font-family:\'Verdana\'; font-size:9pt; font-weight:400; font-style:normal;\">\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">Copyright (c) 2014 Virtual Cable S.L.</span></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'MS Shell Dlg 2\'; font-size:8pt;\"><br /></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">All rights reserved.</span></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'MS Shell Dlg 2\'; font-size:8pt;\"><br /></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">Redistribution and use in source and binary forms, with or without modification,</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">are permitted provided that the following conditions are met:</span></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'MS Shell Dlg 2\'; font-size:8pt;\"><br /></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">   * Redistributions of source code must retain the above copyright notice,</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">     this list of conditions and the following disclaimer.</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">   * Redistributions in binary form must reproduce the above copyright notice,</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">     this list of conditions and the following disclaimer in the documentation</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">     and/or other materials provided with the distribution.</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">   * Neither the name of Virtual Cable S.L. nor the names of its contributors</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">     may be used to endorse or promote products derived from this software</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">     without specific prior written permission.</span></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'MS Shell Dlg 2\'; font-size:8pt;\"><br /></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.</span></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'MS Shell Dlg 2\'; font-size:8pt;\"><br /></p></body></html>", None)) | ||||
|         self.tabWidget.setTabText(self.tabWidget.indexOf(self.licenseTab), _translate("UDSAboutDialog", "&License Agreement", None)) | ||||
|  | ||||
| import UDSActor_rc | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # Form implementation generated from reading ui file 'about-dialog.ui' | ||||
| # | ||||
| # Created: Mon Apr 27 22:05:02 2015 | ||||
| #      by: PyQt4 UI code generator 4.11.2 | ||||
| # | ||||
| # WARNING! All changes made in this file will be lost! | ||||
|  | ||||
| from PyQt4 import QtCore, QtGui | ||||
|  | ||||
| try: | ||||
|     _fromUtf8 = QtCore.QString.fromUtf8 | ||||
| except AttributeError: | ||||
|     def _fromUtf8(s): | ||||
|         return s | ||||
|  | ||||
| try: | ||||
|     _encoding = QtGui.QApplication.UnicodeUTF8 | ||||
|     def _translate(context, text, disambig): | ||||
|         return QtGui.QApplication.translate(context, text, disambig, _encoding) | ||||
| except AttributeError: | ||||
|     def _translate(context, text, disambig): | ||||
|         return QtGui.QApplication.translate(context, text, disambig) | ||||
|  | ||||
| class Ui_UDSAboutDialog(object): | ||||
|     def setupUi(self, UDSAboutDialog): | ||||
|         UDSAboutDialog.setObjectName(_fromUtf8("UDSAboutDialog")) | ||||
|         UDSAboutDialog.resize(466, 402) | ||||
|         font = QtGui.QFont() | ||||
|         font.setFamily(_fromUtf8("Verdana")) | ||||
|         font.setPointSize(9) | ||||
|         UDSAboutDialog.setFont(font) | ||||
|         UDSAboutDialog.setLocale(QtCore.QLocale(QtCore.QLocale.English, QtCore.QLocale.UnitedStates)) | ||||
|         UDSAboutDialog.setModal(True) | ||||
|         self.vboxlayout = QtGui.QVBoxLayout(UDSAboutDialog) | ||||
|         self.vboxlayout.setSpacing(9) | ||||
|         self.vboxlayout.setMargin(9) | ||||
|         self.vboxlayout.setObjectName(_fromUtf8("vboxlayout")) | ||||
|         self.LogoLabel = QtGui.QLabel(UDSAboutDialog) | ||||
|         self.LogoLabel.setObjectName(_fromUtf8("LogoLabel")) | ||||
|         self.vboxlayout.addWidget(self.LogoLabel) | ||||
|         spacerItem = QtGui.QSpacerItem(20, 5, QtGui.QSizePolicy.Minimum, QtGui.QSizePolicy.Fixed) | ||||
|         self.vboxlayout.addItem(spacerItem) | ||||
|         self.TitleLabel = QtGui.QLabel(UDSAboutDialog) | ||||
|         self.TitleLabel.setObjectName(_fromUtf8("TitleLabel")) | ||||
|         self.vboxlayout.addWidget(self.TitleLabel) | ||||
|         self.VersionLabel = QtGui.QLabel(UDSAboutDialog) | ||||
|         self.VersionLabel.setObjectName(_fromUtf8("VersionLabel")) | ||||
|         self.vboxlayout.addWidget(self.VersionLabel) | ||||
|         spacerItem1 = QtGui.QSpacerItem(20, 5, QtGui.QSizePolicy.Minimum, QtGui.QSizePolicy.Fixed) | ||||
|         self.vboxlayout.addItem(spacerItem1) | ||||
|         self.tabWidget = QtGui.QTabWidget(UDSAboutDialog) | ||||
|         self.tabWidget.setObjectName(_fromUtf8("tabWidget")) | ||||
|         self.aboutTab = QtGui.QWidget() | ||||
|         self.aboutTab.setObjectName(_fromUtf8("aboutTab")) | ||||
|         self.vboxlayout1 = QtGui.QVBoxLayout(self.aboutTab) | ||||
|         self.vboxlayout1.setSpacing(6) | ||||
|         self.vboxlayout1.setMargin(9) | ||||
|         self.vboxlayout1.setObjectName(_fromUtf8("vboxlayout1")) | ||||
|         self.aboutBrowser = QtGui.QTextBrowser(self.aboutTab) | ||||
|         self.aboutBrowser.setOpenExternalLinks(True) | ||||
|         self.aboutBrowser.setObjectName(_fromUtf8("aboutBrowser")) | ||||
|         self.vboxlayout1.addWidget(self.aboutBrowser) | ||||
|         self.tabWidget.addTab(self.aboutTab, _fromUtf8("")) | ||||
|         self.authorsTab = QtGui.QWidget() | ||||
|         self.authorsTab.setObjectName(_fromUtf8("authorsTab")) | ||||
|         self.vboxlayout2 = QtGui.QVBoxLayout(self.authorsTab) | ||||
|         self.vboxlayout2.setSpacing(6) | ||||
|         self.vboxlayout2.setMargin(9) | ||||
|         self.vboxlayout2.setObjectName(_fromUtf8("vboxlayout2")) | ||||
|         self.authorsBrowser = QtGui.QTextBrowser(self.authorsTab) | ||||
|         self.authorsBrowser.setOpenExternalLinks(True) | ||||
|         self.authorsBrowser.setObjectName(_fromUtf8("authorsBrowser")) | ||||
|         self.vboxlayout2.addWidget(self.authorsBrowser) | ||||
|         self.tabWidget.addTab(self.authorsTab, _fromUtf8("")) | ||||
|         self.licenseTab = QtGui.QWidget() | ||||
|         self.licenseTab.setObjectName(_fromUtf8("licenseTab")) | ||||
|         self.vboxlayout3 = QtGui.QVBoxLayout(self.licenseTab) | ||||
|         self.vboxlayout3.setSpacing(6) | ||||
|         self.vboxlayout3.setMargin(9) | ||||
|         self.vboxlayout3.setObjectName(_fromUtf8("vboxlayout3")) | ||||
|         self.licenseBrowser = QtGui.QTextBrowser(self.licenseTab) | ||||
|         self.licenseBrowser.setObjectName(_fromUtf8("licenseBrowser")) | ||||
|         self.vboxlayout3.addWidget(self.licenseBrowser) | ||||
|         self.tabWidget.addTab(self.licenseTab, _fromUtf8("")) | ||||
|         self.vboxlayout.addWidget(self.tabWidget) | ||||
|         self.buttonBox = QtGui.QDialogButtonBox(UDSAboutDialog) | ||||
|         self.buttonBox.setOrientation(QtCore.Qt.Horizontal) | ||||
|         self.buttonBox.setStandardButtons(QtGui.QDialogButtonBox.Close) | ||||
|         self.buttonBox.setObjectName(_fromUtf8("buttonBox")) | ||||
|         self.vboxlayout.addWidget(self.buttonBox) | ||||
|  | ||||
|         self.retranslateUi(UDSAboutDialog) | ||||
|         self.tabWidget.setCurrentIndex(0) | ||||
|         QtCore.QObject.connect(self.buttonBox, QtCore.SIGNAL(_fromUtf8("clicked(QAbstractButton*)")), UDSAboutDialog.closeDialog) | ||||
|         QtCore.QMetaObject.connectSlotsByName(UDSAboutDialog) | ||||
|  | ||||
|     def retranslateUi(self, UDSAboutDialog): | ||||
|         UDSAboutDialog.setWindowTitle(_translate("UDSAboutDialog", "About UDS Actor", None)) | ||||
|         self.LogoLabel.setText(_translate("UDSAboutDialog", "<html><head/><body><p><img src=\":/images/img/uds.png\"/> UDS Actor Tools</p></body></html>", None)) | ||||
|         self.TitleLabel.setText(_translate("UDSAboutDialog", "<html><head/><body><p><span style=\" font-family:\'Sans Serif\'; font-size:9pt; font-weight:600;\">UDS Actor</span></p></body></html>", None)) | ||||
|         self.VersionLabel.setText(_translate("UDSAboutDialog", "Version 1.7.0", None)) | ||||
|         self.aboutBrowser.setHtml(_translate("UDSAboutDialog", "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0//EN\" \"http://www.w3.org/TR/REC-html40/strict.dtd\">\n" | ||||
| "<html><head><meta name=\"qrichtext\" content=\"1\" /><style type=\"text/css\">\n" | ||||
| "p, li { white-space: pre-wrap; }\n" | ||||
| "</style></head><body style=\" font-family:\'Verdana\'; font-size:9pt; font-weight:400; font-style:normal;\">\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'Sans Serif\';\"><br /></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'Sans Serif\'; font-weight:600;\">(c) 2014, Virtual Cable S.L.U.</span></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'Sans Serif\'; font-style:italic;\"><br /></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><a href=\"http://www.udsenterprise.com\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt; text-decoration: underline; color:#0000ff;\">http://www.udsenterprise.com</span></a></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><a href=\"http://www.openuds.org\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt; text-decoration: underline; color:#0000ff;\">http://www.openuds.org</span></a></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'Sans Serif\';\"><br /></p></body></html>", None)) | ||||
|         self.tabWidget.setTabText(self.tabWidget.indexOf(self.aboutTab), _translate("UDSAboutDialog", "&About", None)) | ||||
|         self.authorsBrowser.setHtml(_translate("UDSAboutDialog", "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0//EN\" \"http://www.w3.org/TR/REC-html40/strict.dtd\">\n" | ||||
| "<html><head><meta name=\"qrichtext\" content=\"1\" /><style type=\"text/css\">\n" | ||||
| "p, li { white-space: pre-wrap; }\n" | ||||
| "</style></head><body style=\" font-family:\'Verdana\'; font-size:9pt; font-weight:400; font-style:normal;\">\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'Sans Serif\';\">Adolfo Gómez García <agomez@virtualcable.es></span></p></body></html>", None)) | ||||
|         self.tabWidget.setTabText(self.tabWidget.indexOf(self.authorsTab), _translate("UDSAboutDialog", "A&uthors", None)) | ||||
|         self.licenseBrowser.setHtml(_translate("UDSAboutDialog", "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0//EN\" \"http://www.w3.org/TR/REC-html40/strict.dtd\">\n" | ||||
| "<html><head><meta name=\"qrichtext\" content=\"1\" /><style type=\"text/css\">\n" | ||||
| "p, li { white-space: pre-wrap; }\n" | ||||
| "</style></head><body style=\" font-family:\'Verdana\'; font-size:9pt; font-weight:400; font-style:normal;\">\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">Copyright (c) 2014 Virtual Cable S.L.</span></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'MS Shell Dlg 2\'; font-size:8pt;\"><br /></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">All rights reserved.</span></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'MS Shell Dlg 2\'; font-size:8pt;\"><br /></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">Redistribution and use in source and binary forms, with or without modification,</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">are permitted provided that the following conditions are met:</span></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'MS Shell Dlg 2\'; font-size:8pt;\"><br /></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">   * Redistributions of source code must retain the above copyright notice,</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">     this list of conditions and the following disclaimer.</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">   * Redistributions in binary form must reproduce the above copyright notice,</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">     this list of conditions and the following disclaimer in the documentation</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">     and/or other materials provided with the distribution.</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">   * Neither the name of Virtual Cable S.L. nor the names of its contributors</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">     may be used to endorse or promote products derived from this software</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">     without specific prior written permission.</span></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'MS Shell Dlg 2\'; font-size:8pt;\"><br /></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE</span></p>\n" | ||||
| "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:\'MS Shell Dlg 2\'; font-size:8pt;\">OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.</span></p>\n" | ||||
| "<p style=\"-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:\'MS Shell Dlg 2\'; font-size:8pt;\"><br /></p></body></html>", None)) | ||||
|         self.tabWidget.setTabText(self.tabWidget.indexOf(self.licenseTab), _translate("UDSAboutDialog", "&License Agreement", None)) | ||||
|  | ||||
| import UDSActor_rc | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     import sys | ||||
|     app = QtGui.QApplication(sys.argv) | ||||
|     UDSAboutDialog = QtGui.QDialog() | ||||
|     ui = Ui_UDSAboutDialog() | ||||
|     ui.setupUi(UDSAboutDialog) | ||||
|     UDSAboutDialog.show() | ||||
|     sys.exit(app.exec_()) | ||||
|  | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								actors/src/img/uds-512.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 43 KiB | 
| Before Width: | Height: | Size: 3.2 KiB After Width: | Height: | Size: 2.5 KiB | 
| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| # Form implementation generated from reading ui file 'message-dialog.ui' | ||||
| # | ||||
| # Created: Tue Dec  9 11:27:47 2014 | ||||
| # Created: Mon Apr 27 22:05:02 2015 | ||||
| #      by: PyQt4 UI code generator 4.11.2 | ||||
| # | ||||
| # WARNING! All changes made in this file will be lost! | ||||
| @@ -59,3 +59,13 @@ class Ui_UDSMessageDialog(object): | ||||
|     def retranslateUi(self, UDSMessageDialog): | ||||
|         UDSMessageDialog.setWindowTitle(_translate("UDSMessageDialog", "UDS Actor", None)) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     import sys | ||||
|     app = QtGui.QApplication(sys.argv) | ||||
|     UDSMessageDialog = QtGui.QDialog() | ||||
|     ui = Ui_UDSMessageDialog() | ||||
|     ui.setupUi(UDSMessageDialog) | ||||
|     UDSMessageDialog.show() | ||||
|     sys.exit(app.exec_()) | ||||
|  | ||||
|   | ||||
| @@ -121,7 +121,7 @@ | ||||
|      <x>20</x> | ||||
|      <y>20</y> | ||||
|      <width>361</width> | ||||
|      <height>146</height> | ||||
|      <height>131</height> | ||||
|     </rect> | ||||
|    </property> | ||||
|    <layout class="QFormLayout" name="formLayout"> | ||||
| @@ -207,6 +207,9 @@ | ||||
|       <property name="currentIndex"> | ||||
|        <number>1</number> | ||||
|       </property> | ||||
|       <property name="frame"> | ||||
|        <bool>true</bool> | ||||
|       </property> | ||||
|       <item> | ||||
|        <property name="text"> | ||||
|         <string notr="true">DEBUG</string> | ||||
| @@ -219,7 +222,7 @@ | ||||
|       </item> | ||||
|       <item> | ||||
|        <property name="text"> | ||||
|         <string notr="true">EROR</string> | ||||
|         <string notr="true">ERROR</string> | ||||
|        </property> | ||||
|       </item> | ||||
|       <item> | ||||
|   | ||||
| @@ -1,141 +1,152 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # Form implementation generated from reading ui file 'setup-dialog.ui' | ||||
| # | ||||
| # Created: Wed Nov 12 04:50:26 2014 | ||||
| #      by: PyQt4 UI code generator 4.11.2 | ||||
| # | ||||
| # WARNING! All changes made in this file will be lost! | ||||
|  | ||||
| from PyQt4 import QtCore, QtGui | ||||
|  | ||||
| try: | ||||
|     _fromUtf8 = QtCore.QString.fromUtf8 | ||||
| except AttributeError: | ||||
|     def _fromUtf8(s): | ||||
|         return s | ||||
|  | ||||
| try: | ||||
|     _encoding = QtGui.QApplication.UnicodeUTF8 | ||||
|     def _translate(context, text, disambig): | ||||
|         return QtGui.QApplication.translate(context, text, disambig, _encoding) | ||||
| except AttributeError: | ||||
|     def _translate(context, text, disambig): | ||||
|         return QtGui.QApplication.translate(context, text, disambig) | ||||
|  | ||||
| class Ui_UdsActorSetupDialog(object): | ||||
|     def setupUi(self, UdsActorSetupDialog): | ||||
|         UdsActorSetupDialog.setObjectName(_fromUtf8("UdsActorSetupDialog")) | ||||
|         UdsActorSetupDialog.setWindowModality(QtCore.Qt.WindowModal) | ||||
|         UdsActorSetupDialog.resize(400, 243) | ||||
|         sizePolicy = QtGui.QSizePolicy(QtGui.QSizePolicy.Preferred, QtGui.QSizePolicy.Preferred) | ||||
|         sizePolicy.setHorizontalStretch(0) | ||||
|         sizePolicy.setVerticalStretch(0) | ||||
|         sizePolicy.setHeightForWidth(UdsActorSetupDialog.sizePolicy().hasHeightForWidth()) | ||||
|         UdsActorSetupDialog.setSizePolicy(sizePolicy) | ||||
|         font = QtGui.QFont() | ||||
|         font.setFamily(_fromUtf8("Verdana")) | ||||
|         font.setPointSize(9) | ||||
|         UdsActorSetupDialog.setFont(font) | ||||
|         UdsActorSetupDialog.setAutoFillBackground(False) | ||||
|         UdsActorSetupDialog.setLocale(QtCore.QLocale(QtCore.QLocale.English, QtCore.QLocale.UnitedStates)) | ||||
|         UdsActorSetupDialog.setSizeGripEnabled(False) | ||||
|         UdsActorSetupDialog.setModal(True) | ||||
|         self.testButton = QtGui.QPushButton(UdsActorSetupDialog) | ||||
|         self.testButton.setEnabled(False) | ||||
|         self.testButton.setGeometry(QtCore.QRect(20, 160, 361, 23)) | ||||
|         self.testButton.setObjectName(_fromUtf8("testButton")) | ||||
|         self.saveButton = QtGui.QPushButton(UdsActorSetupDialog) | ||||
|         self.saveButton.setEnabled(False) | ||||
|         self.saveButton.setGeometry(QtCore.QRect(20, 190, 101, 23)) | ||||
|         sizePolicy = QtGui.QSizePolicy(QtGui.QSizePolicy.Preferred, QtGui.QSizePolicy.Fixed) | ||||
|         sizePolicy.setHorizontalStretch(0) | ||||
|         sizePolicy.setVerticalStretch(0) | ||||
|         sizePolicy.setHeightForWidth(self.saveButton.sizePolicy().hasHeightForWidth()) | ||||
|         self.saveButton.setSizePolicy(sizePolicy) | ||||
|         self.saveButton.setObjectName(_fromUtf8("saveButton")) | ||||
|         self.cancelButton = QtGui.QPushButton(UdsActorSetupDialog) | ||||
|         self.cancelButton.setGeometry(QtCore.QRect(260, 190, 121, 23)) | ||||
|         sizePolicy = QtGui.QSizePolicy(QtGui.QSizePolicy.Preferred, QtGui.QSizePolicy.Fixed) | ||||
|         sizePolicy.setHorizontalStretch(0) | ||||
|         sizePolicy.setVerticalStretch(0) | ||||
|         sizePolicy.setHeightForWidth(self.cancelButton.sizePolicy().hasHeightForWidth()) | ||||
|         self.cancelButton.setSizePolicy(sizePolicy) | ||||
|         self.cancelButton.setObjectName(_fromUtf8("cancelButton")) | ||||
|         self.layoutWidget = QtGui.QWidget(UdsActorSetupDialog) | ||||
|         self.layoutWidget.setGeometry(QtCore.QRect(20, 20, 361, 146)) | ||||
|         self.layoutWidget.setObjectName(_fromUtf8("layoutWidget")) | ||||
|         self.formLayout = QtGui.QFormLayout(self.layoutWidget) | ||||
|         self.formLayout.setFieldGrowthPolicy(QtGui.QFormLayout.AllNonFixedFieldsGrow) | ||||
|         self.formLayout.setMargin(0) | ||||
|         self.formLayout.setVerticalSpacing(16) | ||||
|         self.formLayout.setObjectName(_fromUtf8("formLayout")) | ||||
|         self.label = QtGui.QLabel(self.layoutWidget) | ||||
|         self.label.setObjectName(_fromUtf8("label")) | ||||
|         self.formLayout.setWidget(0, QtGui.QFormLayout.LabelRole, self.label) | ||||
|         self.host = QtGui.QLineEdit(self.layoutWidget) | ||||
|         self.host.setAcceptDrops(False) | ||||
|         self.host.setObjectName(_fromUtf8("host")) | ||||
|         self.formLayout.setWidget(0, QtGui.QFormLayout.FieldRole, self.host) | ||||
|         self.label_3 = QtGui.QLabel(self.layoutWidget) | ||||
|         self.label_3.setObjectName(_fromUtf8("label_3")) | ||||
|         self.formLayout.setWidget(1, QtGui.QFormLayout.LabelRole, self.label_3) | ||||
|         self.masterKey = QtGui.QLineEdit(self.layoutWidget) | ||||
|         self.masterKey.setObjectName(_fromUtf8("masterKey")) | ||||
|         self.formLayout.setWidget(1, QtGui.QFormLayout.FieldRole, self.masterKey) | ||||
|         self.label_4 = QtGui.QLabel(self.layoutWidget) | ||||
|         self.label_4.setObjectName(_fromUtf8("label_4")) | ||||
|         self.formLayout.setWidget(2, QtGui.QFormLayout.LabelRole, self.label_4) | ||||
|         self.useSSl = QtGui.QComboBox(self.layoutWidget) | ||||
|         self.useSSl.setObjectName(_fromUtf8("useSSl")) | ||||
|         self.useSSl.addItem(_fromUtf8("")) | ||||
|         self.useSSl.addItem(_fromUtf8("")) | ||||
|         self.formLayout.setWidget(2, QtGui.QFormLayout.FieldRole, self.useSSl) | ||||
|         self.logLevelLabel = QtGui.QLabel(self.layoutWidget) | ||||
|         self.logLevelLabel.setObjectName(_fromUtf8("logLevelLabel")) | ||||
|         self.formLayout.setWidget(3, QtGui.QFormLayout.LabelRole, self.logLevelLabel) | ||||
|         self.logLevelComboBox = QtGui.QComboBox(self.layoutWidget) | ||||
|         self.logLevelComboBox.setObjectName(_fromUtf8("logLevelComboBox")) | ||||
|         self.logLevelComboBox.addItem(_fromUtf8("")) | ||||
|         self.logLevelComboBox.setItemText(0, _fromUtf8("DEBUG")) | ||||
|         self.logLevelComboBox.addItem(_fromUtf8("")) | ||||
|         self.logLevelComboBox.setItemText(1, _fromUtf8("INFO")) | ||||
|         self.logLevelComboBox.addItem(_fromUtf8("")) | ||||
|         self.logLevelComboBox.setItemText(2, _fromUtf8("EROR")) | ||||
|         self.logLevelComboBox.addItem(_fromUtf8("")) | ||||
|         self.logLevelComboBox.setItemText(3, _fromUtf8("FATAL")) | ||||
|         self.formLayout.setWidget(3, QtGui.QFormLayout.FieldRole, self.logLevelComboBox) | ||||
|  | ||||
|         self.retranslateUi(UdsActorSetupDialog) | ||||
|         self.logLevelComboBox.setCurrentIndex(1) | ||||
|         QtCore.QObject.connect(self.host, QtCore.SIGNAL(_fromUtf8("textChanged(QString)")), UdsActorSetupDialog.textChanged) | ||||
|         QtCore.QObject.connect(self.masterKey, QtCore.SIGNAL(_fromUtf8("textChanged(QString)")), UdsActorSetupDialog.textChanged) | ||||
|         QtCore.QObject.connect(self.cancelButton, QtCore.SIGNAL(_fromUtf8("pressed()")), UdsActorSetupDialog.cancelAndDiscard) | ||||
|         QtCore.QObject.connect(self.testButton, QtCore.SIGNAL(_fromUtf8("pressed()")), UdsActorSetupDialog.testParameters) | ||||
|         QtCore.QObject.connect(self.saveButton, QtCore.SIGNAL(_fromUtf8("pressed()")), UdsActorSetupDialog.acceptAndSave) | ||||
|         QtCore.QMetaObject.connectSlotsByName(UdsActorSetupDialog) | ||||
|  | ||||
|     def retranslateUi(self, UdsActorSetupDialog): | ||||
|         UdsActorSetupDialog.setWindowTitle(_translate("UdsActorSetupDialog", "UDS Actor Configuration", None)) | ||||
|         self.testButton.setToolTip(_translate("UdsActorSetupDialog", "Click to test the selecter parameters", None)) | ||||
|         self.testButton.setWhatsThis(_translate("UdsActorSetupDialog", "<html><head/><body><p>Click on this button to test the server host and master key parameters.</p><p>A window will be displayed with results after the test is executed.</p><p><br/></p><p>This button will only be active if all parameters are filled.</p></body></html>", None)) | ||||
|         self.testButton.setText(_translate("UdsActorSetupDialog", "Test parameters", None)) | ||||
|         self.saveButton.setToolTip(_translate("UdsActorSetupDialog", "Accepts changes and saves them", None)) | ||||
|         self.saveButton.setWhatsThis(_translate("UdsActorSetupDialog", "Clicking on this button will accept all changes and save them, closing the configuration window", None)) | ||||
|         self.saveButton.setText(_translate("UdsActorSetupDialog", "Accept && Save", None)) | ||||
|         self.cancelButton.setToolTip(_translate("UdsActorSetupDialog", "Cancel all changes and discard them", None)) | ||||
|         self.cancelButton.setWhatsThis(_translate("UdsActorSetupDialog", "Discards all changes and closes the configuration window", None)) | ||||
|         self.cancelButton.setText(_translate("UdsActorSetupDialog", "Cancel && Discard", None)) | ||||
|         self.label.setText(_translate("UdsActorSetupDialog", "UDS Server Host", None)) | ||||
|         self.host.setToolTip(_translate("UdsActorSetupDialog", "Uds Broker Server Addres. Use IP or FQDN", None)) | ||||
|         self.host.setWhatsThis(_translate("UdsActorSetupDialog", "Enter here the UDS Broker Addres using either its IP address or its FQDN address", None)) | ||||
|         self.label_3.setText(_translate("UdsActorSetupDialog", "UDS Master Key", None)) | ||||
|         self.masterKey.setToolTip(_translate("UdsActorSetupDialog", "Master key to communicate with UDS Broker", None)) | ||||
|         self.masterKey.setWhatsThis(_translate("UdsActorSetupDialog", "<html><head/><body><p>Enter the Master Key (found on<span style=\" font-weight:600;\"> UDS Configuration</span> section) of the UDS Broker to allow communication of the Actor with Broker</p></body></html>", None)) | ||||
|         self.label_4.setText(_translate("UdsActorSetupDialog", "Security", None)) | ||||
|         self.useSSl.setToolTip(_translate("UdsActorSetupDialog", "Select communication security with broker", None)) | ||||
|         self.useSSl.setWhatsThis(_translate("UdsActorSetupDialog", "<html><head/><body><p>Select the security for communications with UDS Broker.</p><p>The recommended method of communication is <span style=\" font-weight:600;\">Use SSL</span>, but selection needs to be acording to your broker configuration.</p></body></html>", None)) | ||||
|         self.useSSl.setItemText(0, _translate("UdsActorSetupDialog", "Do not use SSL", None)) | ||||
|         self.useSSl.setItemText(1, _translate("UdsActorSetupDialog", "Use SSL", None)) | ||||
|         self.logLevelLabel.setText(_translate("UdsActorSetupDialog", "Log Level", None)) | ||||
|  | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # Form implementation generated from reading ui file 'setup-dialog.ui' | ||||
| # | ||||
| # Created: Mon Apr 27 22:05:03 2015 | ||||
| #      by: PyQt4 UI code generator 4.11.2 | ||||
| # | ||||
| # WARNING! All changes made in this file will be lost! | ||||
|  | ||||
| from PyQt4 import QtCore, QtGui | ||||
|  | ||||
| try: | ||||
|     _fromUtf8 = QtCore.QString.fromUtf8 | ||||
| except AttributeError: | ||||
|     def _fromUtf8(s): | ||||
|         return s | ||||
|  | ||||
| try: | ||||
|     _encoding = QtGui.QApplication.UnicodeUTF8 | ||||
|     def _translate(context, text, disambig): | ||||
|         return QtGui.QApplication.translate(context, text, disambig, _encoding) | ||||
| except AttributeError: | ||||
|     def _translate(context, text, disambig): | ||||
|         return QtGui.QApplication.translate(context, text, disambig) | ||||
|  | ||||
| class Ui_UdsActorSetupDialog(object): | ||||
|     def setupUi(self, UdsActorSetupDialog): | ||||
|         UdsActorSetupDialog.setObjectName(_fromUtf8("UdsActorSetupDialog")) | ||||
|         UdsActorSetupDialog.setWindowModality(QtCore.Qt.WindowModal) | ||||
|         UdsActorSetupDialog.resize(400, 243) | ||||
|         sizePolicy = QtGui.QSizePolicy(QtGui.QSizePolicy.Preferred, QtGui.QSizePolicy.Preferred) | ||||
|         sizePolicy.setHorizontalStretch(0) | ||||
|         sizePolicy.setVerticalStretch(0) | ||||
|         sizePolicy.setHeightForWidth(UdsActorSetupDialog.sizePolicy().hasHeightForWidth()) | ||||
|         UdsActorSetupDialog.setSizePolicy(sizePolicy) | ||||
|         font = QtGui.QFont() | ||||
|         font.setFamily(_fromUtf8("Verdana")) | ||||
|         font.setPointSize(9) | ||||
|         UdsActorSetupDialog.setFont(font) | ||||
|         UdsActorSetupDialog.setAutoFillBackground(False) | ||||
|         UdsActorSetupDialog.setLocale(QtCore.QLocale(QtCore.QLocale.English, QtCore.QLocale.UnitedStates)) | ||||
|         UdsActorSetupDialog.setSizeGripEnabled(False) | ||||
|         UdsActorSetupDialog.setModal(True) | ||||
|         self.testButton = QtGui.QPushButton(UdsActorSetupDialog) | ||||
|         self.testButton.setEnabled(False) | ||||
|         self.testButton.setGeometry(QtCore.QRect(20, 160, 361, 23)) | ||||
|         self.testButton.setObjectName(_fromUtf8("testButton")) | ||||
|         self.saveButton = QtGui.QPushButton(UdsActorSetupDialog) | ||||
|         self.saveButton.setEnabled(False) | ||||
|         self.saveButton.setGeometry(QtCore.QRect(20, 190, 101, 23)) | ||||
|         sizePolicy = QtGui.QSizePolicy(QtGui.QSizePolicy.Preferred, QtGui.QSizePolicy.Fixed) | ||||
|         sizePolicy.setHorizontalStretch(0) | ||||
|         sizePolicy.setVerticalStretch(0) | ||||
|         sizePolicy.setHeightForWidth(self.saveButton.sizePolicy().hasHeightForWidth()) | ||||
|         self.saveButton.setSizePolicy(sizePolicy) | ||||
|         self.saveButton.setObjectName(_fromUtf8("saveButton")) | ||||
|         self.cancelButton = QtGui.QPushButton(UdsActorSetupDialog) | ||||
|         self.cancelButton.setGeometry(QtCore.QRect(260, 190, 121, 23)) | ||||
|         sizePolicy = QtGui.QSizePolicy(QtGui.QSizePolicy.Preferred, QtGui.QSizePolicy.Fixed) | ||||
|         sizePolicy.setHorizontalStretch(0) | ||||
|         sizePolicy.setVerticalStretch(0) | ||||
|         sizePolicy.setHeightForWidth(self.cancelButton.sizePolicy().hasHeightForWidth()) | ||||
|         self.cancelButton.setSizePolicy(sizePolicy) | ||||
|         self.cancelButton.setObjectName(_fromUtf8("cancelButton")) | ||||
|         self.layoutWidget = QtGui.QWidget(UdsActorSetupDialog) | ||||
|         self.layoutWidget.setGeometry(QtCore.QRect(20, 20, 361, 131)) | ||||
|         self.layoutWidget.setObjectName(_fromUtf8("layoutWidget")) | ||||
|         self.formLayout = QtGui.QFormLayout(self.layoutWidget) | ||||
|         self.formLayout.setFieldGrowthPolicy(QtGui.QFormLayout.AllNonFixedFieldsGrow) | ||||
|         self.formLayout.setMargin(0) | ||||
|         self.formLayout.setVerticalSpacing(16) | ||||
|         self.formLayout.setObjectName(_fromUtf8("formLayout")) | ||||
|         self.label = QtGui.QLabel(self.layoutWidget) | ||||
|         self.label.setObjectName(_fromUtf8("label")) | ||||
|         self.formLayout.setWidget(0, QtGui.QFormLayout.LabelRole, self.label) | ||||
|         self.host = QtGui.QLineEdit(self.layoutWidget) | ||||
|         self.host.setAcceptDrops(False) | ||||
|         self.host.setObjectName(_fromUtf8("host")) | ||||
|         self.formLayout.setWidget(0, QtGui.QFormLayout.FieldRole, self.host) | ||||
|         self.label_3 = QtGui.QLabel(self.layoutWidget) | ||||
|         self.label_3.setObjectName(_fromUtf8("label_3")) | ||||
|         self.formLayout.setWidget(1, QtGui.QFormLayout.LabelRole, self.label_3) | ||||
|         self.masterKey = QtGui.QLineEdit(self.layoutWidget) | ||||
|         self.masterKey.setObjectName(_fromUtf8("masterKey")) | ||||
|         self.formLayout.setWidget(1, QtGui.QFormLayout.FieldRole, self.masterKey) | ||||
|         self.label_4 = QtGui.QLabel(self.layoutWidget) | ||||
|         self.label_4.setObjectName(_fromUtf8("label_4")) | ||||
|         self.formLayout.setWidget(2, QtGui.QFormLayout.LabelRole, self.label_4) | ||||
|         self.useSSl = QtGui.QComboBox(self.layoutWidget) | ||||
|         self.useSSl.setObjectName(_fromUtf8("useSSl")) | ||||
|         self.useSSl.addItem(_fromUtf8("")) | ||||
|         self.useSSl.addItem(_fromUtf8("")) | ||||
|         self.formLayout.setWidget(2, QtGui.QFormLayout.FieldRole, self.useSSl) | ||||
|         self.logLevelLabel = QtGui.QLabel(self.layoutWidget) | ||||
|         self.logLevelLabel.setObjectName(_fromUtf8("logLevelLabel")) | ||||
|         self.formLayout.setWidget(3, QtGui.QFormLayout.LabelRole, self.logLevelLabel) | ||||
|         self.logLevelComboBox = QtGui.QComboBox(self.layoutWidget) | ||||
|         self.logLevelComboBox.setFrame(True) | ||||
|         self.logLevelComboBox.setObjectName(_fromUtf8("logLevelComboBox")) | ||||
|         self.logLevelComboBox.addItem(_fromUtf8("")) | ||||
|         self.logLevelComboBox.setItemText(0, _fromUtf8("DEBUG")) | ||||
|         self.logLevelComboBox.addItem(_fromUtf8("")) | ||||
|         self.logLevelComboBox.setItemText(1, _fromUtf8("INFO")) | ||||
|         self.logLevelComboBox.addItem(_fromUtf8("")) | ||||
|         self.logLevelComboBox.setItemText(2, _fromUtf8("ERROR")) | ||||
|         self.logLevelComboBox.addItem(_fromUtf8("")) | ||||
|         self.logLevelComboBox.setItemText(3, _fromUtf8("FATAL")) | ||||
|         self.formLayout.setWidget(3, QtGui.QFormLayout.FieldRole, self.logLevelComboBox) | ||||
|  | ||||
|         self.retranslateUi(UdsActorSetupDialog) | ||||
|         self.logLevelComboBox.setCurrentIndex(1) | ||||
|         QtCore.QObject.connect(self.host, QtCore.SIGNAL(_fromUtf8("textChanged(QString)")), UdsActorSetupDialog.textChanged) | ||||
|         QtCore.QObject.connect(self.masterKey, QtCore.SIGNAL(_fromUtf8("textChanged(QString)")), UdsActorSetupDialog.textChanged) | ||||
|         QtCore.QObject.connect(self.cancelButton, QtCore.SIGNAL(_fromUtf8("pressed()")), UdsActorSetupDialog.cancelAndDiscard) | ||||
|         QtCore.QObject.connect(self.testButton, QtCore.SIGNAL(_fromUtf8("pressed()")), UdsActorSetupDialog.testParameters) | ||||
|         QtCore.QObject.connect(self.saveButton, QtCore.SIGNAL(_fromUtf8("pressed()")), UdsActorSetupDialog.acceptAndSave) | ||||
|         QtCore.QMetaObject.connectSlotsByName(UdsActorSetupDialog) | ||||
|  | ||||
|     def retranslateUi(self, UdsActorSetupDialog): | ||||
|         UdsActorSetupDialog.setWindowTitle(_translate("UdsActorSetupDialog", "UDS Actor Configuration", None)) | ||||
|         self.testButton.setToolTip(_translate("UdsActorSetupDialog", "Click to test the selecter parameters", None)) | ||||
|         self.testButton.setWhatsThis(_translate("UdsActorSetupDialog", "<html><head/><body><p>Click on this button to test the server host and master key parameters.</p><p>A window will be displayed with results after the test is executed.</p><p><br/></p><p>This button will only be active if all parameters are filled.</p></body></html>", None)) | ||||
|         self.testButton.setText(_translate("UdsActorSetupDialog", "Test parameters", None)) | ||||
|         self.saveButton.setToolTip(_translate("UdsActorSetupDialog", "Accepts changes and saves them", None)) | ||||
|         self.saveButton.setWhatsThis(_translate("UdsActorSetupDialog", "Clicking on this button will accept all changes and save them, closing the configuration window", None)) | ||||
|         self.saveButton.setText(_translate("UdsActorSetupDialog", "Accept && Save", None)) | ||||
|         self.cancelButton.setToolTip(_translate("UdsActorSetupDialog", "Cancel all changes and discard them", None)) | ||||
|         self.cancelButton.setWhatsThis(_translate("UdsActorSetupDialog", "Discards all changes and closes the configuration window", None)) | ||||
|         self.cancelButton.setText(_translate("UdsActorSetupDialog", "Cancel && Discard", None)) | ||||
|         self.label.setText(_translate("UdsActorSetupDialog", "UDS Server Host", None)) | ||||
|         self.host.setToolTip(_translate("UdsActorSetupDialog", "Uds Broker Server Addres. Use IP or FQDN", None)) | ||||
|         self.host.setWhatsThis(_translate("UdsActorSetupDialog", "Enter here the UDS Broker Addres using either its IP address or its FQDN address", None)) | ||||
|         self.label_3.setText(_translate("UdsActorSetupDialog", "UDS Master Key", None)) | ||||
|         self.masterKey.setToolTip(_translate("UdsActorSetupDialog", "Master key to communicate with UDS Broker", None)) | ||||
|         self.masterKey.setWhatsThis(_translate("UdsActorSetupDialog", "<html><head/><body><p>Enter the Master Key (found on<span style=\" font-weight:600;\"> UDS Configuration</span> section) of the UDS Broker to allow communication of the Actor with Broker</p></body></html>", None)) | ||||
|         self.label_4.setText(_translate("UdsActorSetupDialog", "Security", None)) | ||||
|         self.useSSl.setToolTip(_translate("UdsActorSetupDialog", "Select communication security with broker", None)) | ||||
|         self.useSSl.setWhatsThis(_translate("UdsActorSetupDialog", "<html><head/><body><p>Select the security for communications with UDS Broker.</p><p>The recommended method of communication is <span style=\" font-weight:600;\">Use SSL</span>, but selection needs to be acording to your broker configuration.</p></body></html>", None)) | ||||
|         self.useSSl.setItemText(0, _translate("UdsActorSetupDialog", "Do not use SSL", None)) | ||||
|         self.useSSl.setItemText(1, _translate("UdsActorSetupDialog", "Use SSL", None)) | ||||
|         self.logLevelLabel.setText(_translate("UdsActorSetupDialog", "Log Level", None)) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     import sys | ||||
|     app = QtGui.QApplication(sys.argv) | ||||
|     UdsActorSetupDialog = QtGui.QDialog() | ||||
|     ui = Ui_UdsActorSetupDialog() | ||||
|     ui.setupUi(UdsActorSetupDialog) | ||||
|     UdsActorSetupDialog.show() | ||||
|     sys.exit(app.exec_()) | ||||
|  | ||||
|   | ||||
| @@ -39,10 +39,11 @@ import logging | ||||
| import json | ||||
| import uuid | ||||
| import six | ||||
| import warnings | ||||
|  | ||||
| from udsactor.log import logger | ||||
|  | ||||
| from udsactor import __version__ as VERSION | ||||
| from udsactor import VERSION | ||||
| from .utils import exceptionToMessage | ||||
|  | ||||
| VERIFY_CERT = False | ||||
| @@ -81,6 +82,7 @@ except Exception: | ||||
|  | ||||
| try: | ||||
|     urllib3.disable_warnings()  # @UndefinedVariable | ||||
|     warnings.simplefilter("ignore") | ||||
| except Exception: | ||||
|     pass  # In fact, isn't too important, but wil log warns to logging file | ||||
|  | ||||
| @@ -102,15 +104,25 @@ class Api(object): | ||||
|     def __init__(self, host, masterKey, ssl): | ||||
|         self.host = host | ||||
|         self.masterKey = masterKey | ||||
|         self.useSSL = ssl | ||||
|         self.useSSL = True if ssl else False | ||||
|         self.uuid = None | ||||
|         self.mac = None | ||||
|         self.url = "{}://{}/rest/actor/".format(('http', 'https')[ssl], self.host) | ||||
|         self.url = "{}://{}/rest/actor/".format(('http', 'https')[self.useSSL], self.host) | ||||
|         self.idle = None | ||||
|         self.maxSession = None | ||||
|         self.secretKey = six.text_type(uuid.uuid4()) | ||||
|         self.newerRequestLib = requests.__version__.split('.') >= '1' | ||||
|         try: | ||||
|             self.newerRequestLib = requests.__version__.split('.')[0] >= '1' | ||||
|         except Exception: | ||||
|             self.newerRequestLib = False  # I no version, guess this must be an old requests | ||||
|  | ||||
|         # Disable logging requests messages except for errors, ... | ||||
|         logging.getLogger("requests").setLevel(logging.CRITICAL) | ||||
|         # Tries to disable all warnings | ||||
|         try: | ||||
|             warnings.simplefilter("ignore")  # Disables all warnings | ||||
|         except Exception: | ||||
|             pass | ||||
|  | ||||
|     def _getUrl(self, method, key=None, ids=None): | ||||
|         url = self.url + method | ||||
| @@ -129,15 +141,17 @@ class Api(object): | ||||
|     def _request(self, url, data=None): | ||||
|         try: | ||||
|             if data is None: | ||||
|                 # Old requests version does not support verify, but they do not checks ssl certificate | ||||
|                 # Old requests version does not support verify, but they do not checks ssl certificate by default | ||||
|                 if self.newerRequestLib: | ||||
|                     r = requests.get(url, verify=VERIFY_CERT) | ||||
|                 else: | ||||
|                     logger.debug('Requesting with old') | ||||
|                     r = requests.get(url)  # Always ignore certs?? | ||||
|             else: | ||||
|                 if self.newerRequestLib: | ||||
|                     r = requests.post(url, data=data, headers={'content-type': 'application/json'}, verify=VERIFY_CERT) | ||||
|                 else: | ||||
|                     logger.debug('Requesting with old') | ||||
|                     r = requests.post(url, data=data, headers={'content-type': 'application/json'}) | ||||
|  | ||||
|             r = json.loads(r.content)  # Using instead of r.json() to make compatible with oooold rquests lib versions | ||||
| @@ -165,17 +179,24 @@ class Api(object): | ||||
|           uuid, mac | ||||
|           Optionally can return an third parameter, that is max "idle" request time | ||||
|         ''' | ||||
|         logger.debug('Invoking init') | ||||
|         url = self._getUrl('init', key=self.masterKey, ids=ids) | ||||
|         res = self._request(url)['result'] | ||||
|         logger.debug('Got response parameters: {}'.format(res)) | ||||
|         self.uuid, self.mac = res[0:2] | ||||
|         self.idle = int(res[2]) | ||||
|         if self.idle < 15: | ||||
|             self.idle = None  # No values under 30 seconds are allowed :) | ||||
|         # Optional idle parameter | ||||
|         try: | ||||
|             self.idle = int(res[2]) | ||||
|             if self.idle < 30: | ||||
|                 self.idle = None  # No values under 30 seconds are allowed :) | ||||
|         except Exception: | ||||
|             self.idle = None | ||||
|  | ||||
|         return self.uuid | ||||
|  | ||||
|     def postMessage(self, msg, data, processData=True): | ||||
|         logger.debug('Invoking post message {} with data {}'.format(msg, data)) | ||||
|  | ||||
|         if self.uuid is None: | ||||
|             raise ConnectionError('REST api has not been initialized') | ||||
|  | ||||
| @@ -185,22 +206,28 @@ class Api(object): | ||||
|         return self._request(url, data)['result'] | ||||
|  | ||||
|     def notifyComm(self, url): | ||||
|         logger.debug('Notifying comms {}'.format(url)) | ||||
|         return self.postMessage('notifyComms', url) | ||||
|  | ||||
|     def login(self, username): | ||||
|         logger.debug('Notifying login {}'.format(username)) | ||||
|         return self.postMessage('login', username) | ||||
|  | ||||
|     def logout(self, username): | ||||
|         logger.debug('Notifying logout {}'.format(username)) | ||||
|         return self.postMessage('logout', username) | ||||
|  | ||||
|     def information(self): | ||||
|         logger.debug('Requesting information'.format()) | ||||
|         return self.postMessage('information', '') | ||||
|  | ||||
|     def setReady(self, ipsInfo): | ||||
|         logger.debug('Notifying readyness: {}'.format(ipsInfo)) | ||||
|         data = ','.join(['{}={}'.format(v[0], v[1]) for v in ipsInfo]) | ||||
|         return self.postMessage('ready', data) | ||||
|  | ||||
|     def notifyIpChanges(self, ipsInfo): | ||||
|         logger.debug('Notifying ip changes: {}'.format(ipsInfo)) | ||||
|         data = ','.join(['{}={}'.format(v[0], v[1]) for v in ipsInfo]) | ||||
|         return self.postMessage('ip', data) | ||||
|  | ||||
|   | ||||
| @@ -31,17 +31,19 @@ | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| __title__ = 'udsactor' | ||||
| __version__ = '1.7.0' | ||||
| __build__ = 0x010700 | ||||
| __author__ = 'Adolfo Gómez' | ||||
| __license__ = "BSD 3-clause" | ||||
| __copyright__ = "Copyright 2014 VirtualCable S.L.U." | ||||
|  | ||||
|  | ||||
| # On centos, old six release does not includes byte2int, nor six.PY2 | ||||
| import six | ||||
|  | ||||
| VERSION = '2.1.0' | ||||
|  | ||||
| __title__ = 'udsactor' | ||||
| __version__ = VERSION | ||||
| __build__ = 0x010750 | ||||
| __author__ = 'Adolfo Gómez <dkmaster@dkmon.com>' | ||||
| __license__ = "BSD 3-clause" | ||||
| __copyright__ = "Copyright 2014-2016 VirtualCable S.L.U." | ||||
|  | ||||
|  | ||||
| if not hasattr(six, 'byte2int'): | ||||
|     if six.PY3: | ||||
|         import operator | ||||
|   | ||||
| @@ -37,7 +37,8 @@ from udsactor.certs import createSelfSignedCert | ||||
| from udsactor.scriptThread import ScriptExecutorThread | ||||
|  | ||||
| import threading | ||||
| import uuid | ||||
| import string | ||||
| import random | ||||
| import json | ||||
| import six | ||||
| from six.moves import socketserver  # @UnresolvedImport, pylint: disable=import-error | ||||
| @@ -50,6 +51,10 @@ startTime = time.time() | ||||
|  | ||||
|  | ||||
| class HTTPServerHandler(BaseHTTPServer.BaseHTTPRequestHandler): | ||||
|     protocol_version = 'HTTP/1.0' | ||||
|     server_version = 'UDS Actor Server' | ||||
|     sys_version = '' | ||||
|  | ||||
|     uuid = None | ||||
|     service = None | ||||
|     lock = threading.Lock() | ||||
| @@ -61,6 +66,15 @@ class HTTPServerHandler(BaseHTTPServer.BaseHTTPRequestHandler): | ||||
|         self.wfile.write(json.dumps({'error': message})) | ||||
|         return | ||||
|  | ||||
|     def sendJsonResponse(self, data): | ||||
|         self.send_response(200) | ||||
|         data = json.dumps(data) | ||||
|         self.send_header('Content-type', 'application/json') | ||||
|         self.send_header('Content-Length', len(data)) | ||||
|         self.end_headers() | ||||
|         # Send the html message | ||||
|         self.wfile.write(data) | ||||
|  | ||||
|     def do_GET(self): | ||||
|         # Very simple path & params splitter | ||||
|         path = self.path.split('?')[0][1:].split('/') | ||||
| @@ -74,11 +88,7 @@ class HTTPServerHandler(BaseHTTPServer.BaseHTTPRequestHandler): | ||||
|             return | ||||
|  | ||||
|         if len(path) != 2: | ||||
|             self.send_response(200) | ||||
|             self.send_header('Content-type', 'application/json') | ||||
|             self.end_headers() | ||||
|             # Send the html message | ||||
|             self.wfile.write(json.dumps("UDS Actor has been running for {} seconds".format(time.time() - startTime))) | ||||
|             self.sendJsonResponse("UDS Actor has been running for {} seconds".format(time.time() - startTime)) | ||||
|             return | ||||
|  | ||||
|         try: | ||||
| @@ -92,11 +102,7 @@ class HTTPServerHandler(BaseHTTPServer.BaseHTTPRequestHandler): | ||||
|             self.sendJsonError(500, str(e)) | ||||
|             return | ||||
|  | ||||
|         self.send_response(200) | ||||
|         self.send_header('Content-type', 'application/json') | ||||
|         self.end_headers() | ||||
|         # Send the html message | ||||
|         self.wfile.write(json.dumps(result)) | ||||
|         self.sendJsonResponse(result) | ||||
|  | ||||
|     def do_POST(self): | ||||
|         path = self.path.split('?')[0][1:].split('/') | ||||
| @@ -112,7 +118,7 @@ class HTTPServerHandler(BaseHTTPServer.BaseHTTPRequestHandler): | ||||
|             HTTPServerHandler.lock.acquire() | ||||
|             length = int(self.headers.getheader('content-length')) | ||||
|             content = self.rfile.read(length) | ||||
|             print(length, ">>", content, '<<') | ||||
|             logger.debug('length: {}, content >>{}<<'.format(length, content)) | ||||
|             params = json.loads(content) | ||||
|  | ||||
|             operation = getattr(self, 'post_' + path[1]) | ||||
| @@ -127,11 +133,7 @@ class HTTPServerHandler(BaseHTTPServer.BaseHTTPRequestHandler): | ||||
|         finally: | ||||
|             HTTPServerHandler.lock.release() | ||||
|  | ||||
|         self.send_response(200) | ||||
|         self.send_header('Content-type', 'application/json') | ||||
|         self.end_headers() | ||||
|         # Send the html message | ||||
|         self.wfile.write(json.dumps(result)) | ||||
|         self.sendJsonResponse(result) | ||||
|  | ||||
|     def post_logoff(self, params): | ||||
|         logger.debug('Sending LOGOFF to clients') | ||||
| @@ -170,7 +172,16 @@ class HTTPServerHandler(BaseHTTPServer.BaseHTTPRequestHandler): | ||||
|  | ||||
|     def get_information(self, params): | ||||
|         # TODO: Return something useful? :) | ||||
|         return 'Information' | ||||
|         return 'Up and running' | ||||
|  | ||||
|     def get_uuid(self, params): | ||||
|         return self.service.api.uuid | ||||
|  | ||||
|     def log_error(self, fmt, *args): | ||||
|         logger.error('HTTP ' + fmt % args) | ||||
|  | ||||
|     def log_message(self, fmt, *args): | ||||
|         logger.info('HTTP ' + fmt % args) | ||||
|  | ||||
|  | ||||
| class HTTPServerThread(threading.Thread): | ||||
| @@ -178,20 +189,45 @@ class HTTPServerThread(threading.Thread): | ||||
|         super(self.__class__, self).__init__() | ||||
|  | ||||
|         if HTTPServerHandler.uuid is None: | ||||
|             HTTPServerHandler.uuid = uuid.uuid4().get_hex() | ||||
|  | ||||
|         HTTPServerHandler.service = service | ||||
|             HTTPServerHandler.uuid = ''.join(random.choice(string.ascii_letters + string.digits) for _ in range(48)) | ||||
|  | ||||
|         self.certFile = createSelfSignedCert() | ||||
|         self.server = socketserver.TCPServer(address, HTTPServerHandler) | ||||
|         HTTPServerHandler.service = service | ||||
|  | ||||
|         self.initiateServer(address) | ||||
|  | ||||
|     def getPort(self): | ||||
|         return self.address[1] | ||||
|  | ||||
|     def getIp(self): | ||||
|         return self.address[0] | ||||
|  | ||||
|     def initiateServer(self, address): | ||||
|         self.address = (address[0], address[1])  # Copy address & keep it for future reference... | ||||
|  | ||||
|         addr = ('0.0.0.0', address[1])  # Adapt to listen on 0.0.0.0 | ||||
|  | ||||
|         self.server = socketserver.TCPServer(addr, HTTPServerHandler) | ||||
|         self.server.socket = ssl.wrap_socket(self.server.socket, certfile=self.certFile, server_side=True) | ||||
|  | ||||
|     def getServerUrl(self): | ||||
|         return 'https://{}:{}/{}'.format(self.server.server_address[0], self.server.server_address[1], HTTPServerHandler.uuid) | ||||
|         return 'https://{}:{}/{}'.format(self.getIp(), self.getPort(), HTTPServerHandler.uuid) | ||||
|  | ||||
|     def stop(self): | ||||
|         logger.debug('Stopping REST Service') | ||||
|         self.server.shutdown() | ||||
|  | ||||
|     def restart(self, address=None): | ||||
|  | ||||
|         if address is None: | ||||
|             # address = self.server.server_address | ||||
|             address = self.address | ||||
|  | ||||
|         self.address = (address[0], self.address[1])  # Copy address & keep it for future reference, port is never changed once assigned on init | ||||
|  | ||||
|         # Listening on 0.0.0.0, does not need to restart listener.. | ||||
|         # self.stop() | ||||
|         # self.initiateServer(address) | ||||
|  | ||||
|     def run(self): | ||||
|         self.server.serve_forever() | ||||
|   | ||||
| @@ -36,6 +36,8 @@ import sys | ||||
| import six | ||||
| import traceback | ||||
| import pickle | ||||
| import errno | ||||
| import time | ||||
|  | ||||
| from udsactor.utils import toUnicode | ||||
| from udsactor.log import logger | ||||
| @@ -407,8 +409,11 @@ class ClientIPC(threading.Thread): | ||||
|                 self.messageReceived() | ||||
|  | ||||
|             except socket.error as e: | ||||
|                 if e.errno == errno.EINTR: | ||||
|                     time.sleep(1)  # | ||||
|                     continue  # Ignore interrupted system call | ||||
|                 logger.error('Communication with server got an error: {}'.format(toUnicode(e.strerror))) | ||||
|                 self.running = False | ||||
|                 # self.running = False | ||||
|                 return | ||||
|             except Exception as e: | ||||
|                 tb = traceback.format_exc() | ||||
|   | ||||
| @@ -36,6 +36,7 @@ from udsactor import operations | ||||
| from udsactor.service import CommonService | ||||
| from udsactor.service import initCfg | ||||
| from udsactor.service import IPC_PORT | ||||
|  | ||||
| from udsactor import ipc | ||||
|  | ||||
| from udsactor.log import logger | ||||
| @@ -44,15 +45,23 @@ from udsactor.linux.daemon import Daemon | ||||
| from udsactor.linux import renamer | ||||
|  | ||||
| import sys | ||||
| import time | ||||
| import os | ||||
| import stat | ||||
| import subprocess | ||||
|  | ||||
| POST_CMD = '/etc/udsactor/post' | ||||
| PRECONNECT_CMD = '/etc/udsactor/pre' | ||||
|  | ||||
| try: | ||||
|     from prctl import set_proctitle | ||||
|     from prctl import set_proctitle  # @UnresolvedImport | ||||
| except Exception:  # Platform may not include prctl, so in case it's not available, we let the "name" as is | ||||
|     def set_proctitle(_): | ||||
|         pass | ||||
|  | ||||
|  | ||||
| class UDSActorSvc(Daemon, CommonService): | ||||
|     rebootMachineAfterOp = False | ||||
|  | ||||
|     def __init__(self, args=None): | ||||
|         Daemon.__init__(self, '/var/run/udsa.pid') | ||||
|         CommonService.__init__(self) | ||||
| @@ -62,6 +71,12 @@ class UDSActorSvc(Daemon, CommonService): | ||||
|         Renames the computer, and optionally sets a password for an user | ||||
|         before this | ||||
|         ''' | ||||
|         hostName = operations.getComputerName() | ||||
|  | ||||
|         if hostName.lower() == name.lower(): | ||||
|             logger.info('Computer name is already {}'.format(hostName)) | ||||
|             self.setReady() | ||||
|             return | ||||
|  | ||||
|         # Check for password change request for an user | ||||
|         if user is not None: | ||||
| @@ -75,21 +90,63 @@ class UDSActorSvc(Daemon, CommonService): | ||||
|                     'Could not change password for user {} (maybe invalid current password is configured at broker): {} '.format(user, unicode(e))) | ||||
|  | ||||
|         renamer.rename(name) | ||||
|         self.setReady() | ||||
|  | ||||
|         if self.rebootMachineAfterOp is False: | ||||
|             self.setReady() | ||||
|         else: | ||||
|             logger.info('Rebooting computer to activate new name {}'.format(name)) | ||||
|             self.reboot() | ||||
|  | ||||
|  | ||||
|     def joinDomain(self, name, domain, ou, account, password): | ||||
|         logger.fatal('Join domain is not supported on linux platforms right now') | ||||
|  | ||||
|     def preConnect(self, user, protocol): | ||||
|         ''' | ||||
|         Invoked when received a PRE Connection request via REST | ||||
|         ''' | ||||
|         # Execute script in /etc/udsactor/post after interacting with broker, if no reboot is requested ofc | ||||
|         # This will be executed only when machine gets "ready" | ||||
|         try: | ||||
|  | ||||
|             if os.path.isfile(PRECONNECT_CMD): | ||||
|                 if (os.stat(PRECONNECT_CMD).st_mode & stat.S_IXUSR) != 0: | ||||
|                     subprocess.call([PRECONNECT_CMD, user, protocol]) | ||||
|                 else: | ||||
|                     logger.info('PRECONNECT file exists but it it is not executable (needs execution permission by root)') | ||||
|             else: | ||||
|                 logger.info('PRECONNECT file not found & not executed') | ||||
|         except Exception as e: | ||||
|             # Ignore output of execution command | ||||
|             logger.error('Executing preconnect command give') | ||||
|  | ||||
|  | ||||
|     def run(self): | ||||
|         initCfg() | ||||
|         cfg = initCfg()  # Gets a local copy of config to get "reboot" | ||||
|  | ||||
|         logger.debug('CFG: {}'.format(cfg)) | ||||
|  | ||||
|         if cfg is not None: | ||||
|             self.rebootMachineAfterOp = cfg.get('reboot', True) | ||||
|         else: | ||||
|             self.rebootMachineAfterOp = False | ||||
|  | ||||
|         logger.info('Reboot after is {}'.format(self.rebootMachineAfterOp)) | ||||
|  | ||||
|         logger.debug('Running Daemon') | ||||
|         set_proctitle('UDSActorDaemon') | ||||
|  | ||||
|         # Linux daemon will continue running unless something is requested to | ||||
|         if self.interactWithBroker() is False: | ||||
|             logger.debug('Interact with broker returned false, stopping service after a while') | ||||
|             return | ||||
|         while True: | ||||
|             brokerConnected = self.interactWithBroker() | ||||
|             if brokerConnected is False: | ||||
|                 logger.debug('Interact with broker returned false, stopping service after a while') | ||||
|                 return | ||||
|             elif brokerConnected is True: | ||||
|                 break | ||||
|  | ||||
|             # If brokerConnected returns None, repeat the cycle | ||||
|             self.doWait(16000)  # Wait for a looong while | ||||
|  | ||||
|         if self.isAlive is False: | ||||
|             logger.debug('The service is not alive after broker interaction, stopping it') | ||||
| @@ -99,6 +156,21 @@ class UDSActorSvc(Daemon, CommonService): | ||||
|             logger.debug('Reboot has been requested, stopping service') | ||||
|             return | ||||
|  | ||||
|         # Execute script in /etc/udsactor/post after interacting with broker, if no reboot is requested ofc | ||||
|         # This will be executed only when machine gets "ready" | ||||
|         try: | ||||
|  | ||||
|             if os.path.isfile(POST_CMD): | ||||
|                 if (os.stat(POST_CMD).st_mode & stat.S_IXUSR) != 0: | ||||
|                     subprocess.call([POST_CMD, ]) | ||||
|                 else: | ||||
|                     logger.info('POST file exists but it it is not executable (needs execution permission by root)') | ||||
|             else: | ||||
|                 logger.info('POST file not found & not executed') | ||||
|         except Exception as e: | ||||
|             # Ignore output of execution command | ||||
|             logger.error('Executing post command give') | ||||
|  | ||||
|         self.initIPC() | ||||
|  | ||||
|         # ********************* | ||||
|   | ||||
| @@ -57,7 +57,7 @@ def _getMacAddr(ifname): | ||||
|     try: | ||||
|         s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) | ||||
|         info = bytearray(fcntl.ioctl(s.fileno(), 0x8927, struct.pack(str('256s'), ifname[:15]))) | ||||
|         return six.text_type(''.join(['%02x:' % char for char in info[18:24]])[:-1]) | ||||
|         return six.text_type(''.join(['%02x:' % char for char in info[18:24]])[:-1]).upper() | ||||
|     except Exception: | ||||
|         return None | ||||
|  | ||||
| @@ -122,7 +122,7 @@ def getComputerName(): | ||||
| def getNetworkInfo(): | ||||
|     for ifname in _getInterfaces(): | ||||
|         ip, mac = _getIpAndMac(ifname) | ||||
|         if mac != '00:00:00:00:00:00':  # Skips local interfaces | ||||
|         if mac != '00:00:00:00:00:00' and ip.startswith('169.254') is False:  # Skips local interfaces & interfaces with no dhcp IPs | ||||
|             yield utils.Bunch(name=ifname, mac=mac, ip=ip) | ||||
|  | ||||
|  | ||||
| @@ -194,8 +194,10 @@ try: | ||||
|     # Fix result type to XScreenSaverInfo Structure | ||||
|     xss.XScreenSaverQueryExtension.restype = ctypes.c_int | ||||
|     xss.XScreenSaverAllocInfo.restype = ctypes.POINTER(XScreenSaverInfo)  # Result in a XScreenSaverInfo structure | ||||
|     display = xlib.XOpenDisplay(None) | ||||
|     info = xss.XScreenSaverAllocInfo() | ||||
| except Exception:  # Libraries not accesible, not found or whatever.. | ||||
|     xlib = xss = None | ||||
|     xlib = xss = display = info = None | ||||
|  | ||||
|  | ||||
| def initIdleDuration(atLeastSeconds): | ||||
| @@ -208,6 +210,8 @@ def initIdleDuration(atLeastSeconds): | ||||
|         threading._DummyThread._Thread__stop = lambda x: 42 | ||||
|  | ||||
|     subprocess.call(['/usr/bin/xset', 's', '{}'.format(atLeastSeconds + 30)]) | ||||
|     # And now reset it | ||||
|     subprocess.call(['/usr/bin/xset', 's', 'reset']) | ||||
|  | ||||
|  | ||||
| def getIdleDuration(): | ||||
| @@ -217,18 +221,20 @@ def getIdleDuration(): | ||||
|     if xlib is None or xss is None: | ||||
|         return 0  # Libraries not available | ||||
|  | ||||
|     # production code might want to not hardcode the offset 16... | ||||
|     display = xlib.XOpenDisplay(None) | ||||
|  | ||||
|     event_base = ctypes.c_int() | ||||
|     error_base = ctypes.c_int() | ||||
|  | ||||
|     available = xss.XScreenSaverQueryExtension(display, ctypes.byref(event_base), ctypes.byref(error_base)) | ||||
|     if available != 1: | ||||
|         return 0 | ||||
|  | ||||
|     info = xss.XScreenSaverAllocInfo() | ||||
|     if available != 1: | ||||
|         return 0  # No screen saver is available, no way of getting idle | ||||
|  | ||||
|     xss.XScreenSaverQueryInfo(display, xlib.XDefaultRootWindow(display), info) | ||||
|  | ||||
|     # Centos seems to set state to 1?? (weird, but it's happening don't know why... will try this way) | ||||
|     if info.contents.state != 0 and 'centos' not in platform.linux_distribution()[0].lower().strip(): | ||||
|         return 3600 * 100 * 1000  # If screen saver is active, return a high enough value | ||||
|  | ||||
|     return info.contents.idle / 1000.0 | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -78,3 +78,11 @@ def writeConfig(data): | ||||
|         cfg.write(f) | ||||
|  | ||||
|     os.chmod(CONFIGFILE, 0o0600) | ||||
|  | ||||
| def useOldJoinSystem(): | ||||
|     return False | ||||
|  | ||||
| # Right now, we do not really need an application to be run on "startup" as could ocur with windows | ||||
| def runApplication(): | ||||
|     return None | ||||
|  | ||||
|   | ||||
| @@ -31,6 +31,7 @@ | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import traceback | ||||
| import sys | ||||
| import six | ||||
|  | ||||
| @@ -50,6 +51,10 @@ class Logger(object): | ||||
|         self.remoteLogger = None | ||||
|  | ||||
|     def setLevel(self, level): | ||||
|         ''' | ||||
|         Sets log level filter (minimum level required for a log message to be processed) | ||||
|         :param level: Any message with a level below this will be filtered out | ||||
|         ''' | ||||
|         self.logLevel = int(level)  # Ensures level is an integer or fails | ||||
|  | ||||
|     def setRemoteLogger(self, remoteLogger): | ||||
| @@ -75,7 +80,7 @@ class Logger(object): | ||||
|         self.log(WARN, message) | ||||
|  | ||||
|     def info(self, message): | ||||
|         self.log(WARN, message) | ||||
|         self.log(INFO, message) | ||||
|  | ||||
|     def error(self, message): | ||||
|         self.log(ERROR, message) | ||||
| @@ -83,6 +88,14 @@ class Logger(object): | ||||
|     def fatal(self, message): | ||||
|         self.log(FATAL, message) | ||||
|  | ||||
|     def exception(self): | ||||
|         try: | ||||
|             tb = traceback.format_exc() | ||||
|         except Exception: | ||||
|             tb = '(could not get traceback!)' | ||||
|  | ||||
|         self.log(DEBUG, tb) | ||||
|  | ||||
|     def flush(self): | ||||
|         pass | ||||
|  | ||||
|   | ||||
| @@ -70,6 +70,8 @@ def initCfg(): | ||||
|             cfg = None | ||||
|             break | ||||
|  | ||||
|     return cfg | ||||
|  | ||||
|  | ||||
| class CommonService(object): | ||||
|     def __init__(self): | ||||
| @@ -84,6 +86,23 @@ class CommonService(object): | ||||
|     def reboot(self): | ||||
|         self.rebootRequested = True | ||||
|  | ||||
|     def execute(self, cmd, section): | ||||
|         import os | ||||
|         import subprocess | ||||
|         import stat | ||||
|  | ||||
|         if os.path.isfile(cmd): | ||||
|             if (os.stat(cmd).st_mode & stat.S_IXUSR) != 0: | ||||
|                 subprocess.call([cmd, ]) | ||||
|                 return True | ||||
|             else: | ||||
|                 logger.info('{} file exists but it it is not executable (needs execution permission by admin/root)'.format(section)) | ||||
|         else: | ||||
|             logger.info('{} file not found & not executed'.format(section)) | ||||
|  | ||||
|         return False | ||||
|  | ||||
|  | ||||
|     def setReady(self): | ||||
|         self.api.setReady([(v.mac, v.ip) for v in operations.getNetworkInfo()]) | ||||
|  | ||||
| @@ -108,7 +127,7 @@ class CommonService(object): | ||||
|                 ids = ','.join([i.mac for i in netInfo]) | ||||
|                 if ids == '': | ||||
|                     # Wait for any network interface to be ready | ||||
|                     logger.debug('No network interfaces found, retrying in a while...') | ||||
|                     logger.debug('No valid network interfaces found, retrying in a while...') | ||||
|                     raise Exception() | ||||
|                 logger.debug('Ids: {}'.format(ids)) | ||||
|                 self.api.init(ids) | ||||
| @@ -126,7 +145,7 @@ class CommonService(object): | ||||
|                 logger.fatal('This host is not managed by UDS Broker (ids: {})'.format(ids)) | ||||
|                 return False  # On unmanaged hosts, there is no reason right now to continue running | ||||
|             except Exception as e: | ||||
|                 logger.debug('Exception caught: {}, retrying'.format(exceptionToMessage(e))) | ||||
|                 logger.debug('Exception on network info: retrying') | ||||
|                 # Any other error is expectable and recoverable, so let's wait a bit and retry again | ||||
|                 # but, if too many errors, will log it (one every minute, for | ||||
|                 # example) | ||||
| @@ -136,6 +155,13 @@ class CommonService(object): | ||||
|                 # Wait a bit before next check | ||||
|                 self.doWait(5000) | ||||
|  | ||||
|         # Now try to run the "runonce" element | ||||
|         runOnce = store.runApplication() | ||||
|         if runOnce is not None: | ||||
|             if self.execute(runOnce, 'RunOnce') is True: | ||||
|                 # operations.reboot() | ||||
|                 return False | ||||
|  | ||||
|         # Broker connection is initialized, now get information about what to | ||||
|         # do | ||||
|         counter = 0 | ||||
| @@ -151,9 +177,11 @@ class CommonService(object): | ||||
|                 if data[0] == 'rename': | ||||
|                     try: | ||||
|                         if len(params) == 1:  # Simple rename | ||||
|                             logger.debug('Renaming computer to {}'.format(params[0])) | ||||
|                             self.rename(params[0]) | ||||
|                         # Rename with change password for an user | ||||
|                         elif len(params) == 4: | ||||
|                             logger.debug('Renaming computer to {}'.format(params)) | ||||
|                             self.rename(params[0], params[1], params[2], params[3]) | ||||
|                         else: | ||||
|                             logger.error('Got invalid parameter for rename operation: {}'.format(params)) | ||||
| @@ -161,11 +189,11 @@ class CommonService(object): | ||||
|                         break | ||||
|                     except Exception as e: | ||||
|                         logger.error('Error at computer renaming stage: {}'.format(e.message)) | ||||
|                         return False | ||||
|                         return None  # Will retry complete broker connection if this point is reached | ||||
|                 elif data[0] == 'domain': | ||||
|                     if len(params) != 5: | ||||
|                         logger.error('Got invalid parameters for domain message: {}'.format(params)) | ||||
|                         return False | ||||
|                         return False  # Stop running service | ||||
|                     self.joinDomain(params[0], params[1], params[2], params[3], params[4]) | ||||
|                     break | ||||
|                 else: | ||||
| @@ -174,10 +202,10 @@ class CommonService(object): | ||||
|             except REST.UserServiceNotFoundError: | ||||
|                 logger.error('The host has lost the sync state with broker! (host uuid changed?)') | ||||
|                 return False | ||||
|             except Exception: | ||||
|                 counter += 1 | ||||
|             except Exception as err: | ||||
|                 if counter % 60 == 0: | ||||
|                     logger.warn('Too many retries in progress, though still trying (last error: {})'.format(exceptionToMessage(e))) | ||||
|                     logger.warn('Too many retries in progress, though still trying (last error: {})'.format(exceptionToMessage(err))) | ||||
|                 counter += 1 | ||||
|                 # Any other error is expectable and recoverable, so let's wait | ||||
|                 # a bit and retry again | ||||
|                 # Wait a bit before next check | ||||
| @@ -193,7 +221,7 @@ class CommonService(object): | ||||
|         return True | ||||
|  | ||||
|     def checkIpsChanged(self): | ||||
|         if self.api.uuid is None: | ||||
|         if self.api is None or self.api.uuid is None: | ||||
|             return  # Not connected | ||||
|         netInfo = tuple(operations.getNetworkInfo()) | ||||
|         for i in netInfo: | ||||
| @@ -203,8 +231,17 @@ class CommonService(object): | ||||
|                 try: | ||||
|                     # Notifies all interfaces IPs | ||||
|                     self.api.notifyIpChanges(((v.mac, v.ip) for v in netInfo)) | ||||
|  | ||||
|                     # Regenerates Known ips | ||||
|                     self.knownIps = dict(((i.mac, i.ip) for i in netInfo)) | ||||
|                     self.knownIps = dict(((v.mac, v.ip) for v in netInfo)) | ||||
|  | ||||
|                     # And notify new listening address to broker | ||||
|                     address = (self.knownIps[self.api.mac], self.httpServer.getPort()) | ||||
|                     # And new listening address | ||||
|                     self.httpServer.restart(address) | ||||
|                     # sends notification | ||||
|                     self.api.notifyComm(self.httpServer.getServerUrl()) | ||||
|  | ||||
|                 except Exception as e: | ||||
|                     logger.warn('Got an error notifiying IPs to broker: {} (will retry in a bit)'.format(e.message.decode('windows-1250', 'ignore'))) | ||||
|  | ||||
| @@ -215,14 +252,21 @@ class CommonService(object): | ||||
|             return | ||||
|  | ||||
|         if msg == ipc.REQ_LOGIN: | ||||
|             self.api.login(data) | ||||
|         elif msg == ipc.REQ_LOGOUT: | ||||
|             res = self.api.login(data).split('\t') | ||||
|             # third parameter, if exists, sets maxSession duration to this. | ||||
|             # First & second parameters are ip & hostname of connection source | ||||
|             if len(res) >= 3: | ||||
|                 self.api.maxSession = int(res[2])  # Third parameter is max session duration | ||||
|                 msg = ipc.REQ_INFORMATION  # Senf information, requested or not, to client on login notification | ||||
|         if msg == ipc.REQ_LOGOUT: | ||||
|             self.api.logout(data) | ||||
|             self.onLogout(data) | ||||
|         elif msg == ipc.REQ_INFORMATION: | ||||
|         if msg == ipc.REQ_INFORMATION: | ||||
|             info = {} | ||||
|             if self.api.idle is not None: | ||||
|                 info['idle'] = self.api.idle | ||||
|             if self.api.maxSession is not None: | ||||
|                 info['maxSession'] = self.api.maxSession | ||||
|             self.ipc.sendInformationMessage(info) | ||||
|  | ||||
|     def initIPC(self): | ||||
| @@ -234,8 +278,8 @@ class CommonService(object): | ||||
|         self.ipc.start() | ||||
|  | ||||
|         if self.api.mac in self.knownIps: | ||||
|             address = (self.knownIps[self.api.mac], random.randrange(40000, 44000)) | ||||
|             logger.debug('Starting REST listener at {}'.format(address)) | ||||
|             address = (self.knownIps[self.api.mac], random.randrange(43900, 44000)) | ||||
|             logger.info('Starting REST listener at {}'.format(address)) | ||||
|             self.httpServer = httpserver.HTTPServerThread(address, self) | ||||
|             self.httpServer.start() | ||||
|             # And notify it to broker | ||||
|   | ||||
| @@ -76,7 +76,7 @@ class SensLogon(win32com.server.policy.DesignatedWrapPolicy): | ||||
|                 data = self.service.api.login(args[0]) | ||||
|                 logger.debug('Data received for login: {}'.format(data)) | ||||
|                 data = data.split('\t') | ||||
|                 if len(data) == 2: | ||||
|                 if len(data) >= 2: | ||||
|                     logger.debug('Data is valid: {}'.format(data)) | ||||
|                     windir = os.environ['windir'] | ||||
|                     with open(os.path.join(windir, 'remoteip.txt'), 'w') as f: | ||||
|   | ||||
| @@ -40,8 +40,11 @@ import win32event  # @UnresolvedImport, pylint: disable=import-error | ||||
| import win32com.client  # @UnresolvedImport,  @UnusedImport, pylint: disable=import-error | ||||
| import pythoncom  # @UnresolvedImport, pylint: disable=import-error | ||||
| import servicemanager  # @UnresolvedImport, pylint: disable=import-error | ||||
| import subprocess | ||||
| import os | ||||
|  | ||||
| from udsactor import operations | ||||
| from udsactor import store | ||||
| from udsactor.service import CommonService | ||||
| from udsactor.service import initCfg | ||||
|  | ||||
| @@ -54,6 +57,8 @@ from .SENS import SENSGUID_PUBLISHER | ||||
| from .SENS import PROGID_EventSubscription | ||||
| from .SENS import PROGID_EventSystem | ||||
|  | ||||
| POST_CMD = 'c:\\windows\\post-uds.bat' | ||||
|  | ||||
|  | ||||
| class UDSActorSvc(win32serviceutil.ServiceFramework, CommonService): | ||||
|     ''' | ||||
| @@ -112,7 +117,7 @@ class UDSActorSvc(win32serviceutil.ServiceFramework, CommonService): | ||||
|  | ||||
|         operations.renameComputer(name) | ||||
|         # Reboot just after renaming | ||||
|         logger.info('Rebooting computer got activate new name {}'.format(name)) | ||||
|         logger.info('Rebooting computer to activate new name {}'.format(name)) | ||||
|         self.reboot() | ||||
|  | ||||
|     def oneStepJoin(self, name, domain, ou, account, password): | ||||
| @@ -137,13 +142,15 @@ class UDSActorSvc(win32serviceutil.ServiceFramework, CommonService): | ||||
|         currName = operations.getComputerName() | ||||
|         if currName.lower() == name.lower(): | ||||
|             currDomain = operations.getDomainName() | ||||
|             if currDomain is not None and currDomain.lower() == domain.lower(): | ||||
|             if currDomain is not None: | ||||
|                 # logger.debug('Name: "{}" vs "{}", Domain: "{}" vs "{}"'.format(currName.lower(), name.lower(), currDomain.lower(), domain.lower())) | ||||
|                 logger.info( | ||||
|                     'Machine {} is part of domain {}'.format(name, domain)) | ||||
|                 self.setReady() | ||||
|             else: | ||||
|                 operations.joinDomain( | ||||
|                     domain, ou, account, password, executeInOneStep=False) | ||||
|                 self.reboot() | ||||
|         else: | ||||
|             operations.renameComputer(name) | ||||
|             logger.info( | ||||
| @@ -153,17 +160,21 @@ class UDSActorSvc(win32serviceutil.ServiceFramework, CommonService): | ||||
|     def joinDomain(self, name, domain, ou, account, password): | ||||
|         ver = operations.getWindowsVersion() | ||||
|         ver = ver[0] * 10 + ver[1] | ||||
|         logger.info('Starting joining domain {} with name {} (detected operating version: {})'.format( | ||||
|         logger.debug('Starting joining domain {} with name {} (detected operating version: {})'.format( | ||||
|             domain, name, ver)) | ||||
|         # If file c:\compat.bin exists, joind domain in two steps instead one | ||||
|  | ||||
|         # Accepts one step joinDomain, also remember XP is no more supported by | ||||
|         # microsoft, but this also must works with it because will do a "multi | ||||
|         # step" join | ||||
|         if ver >= 60: | ||||
|         if ver >= 60 and store.useOldJoinSystem() is False: | ||||
|             self.oneStepJoin(name, domain, ou, account, password) | ||||
|         else: | ||||
|             logger.info('Using multiple step join because configuration requests to do so') | ||||
|             self.multiStepJoin(name, domain, ou, account, password) | ||||
|  | ||||
|     def preConnect(self, user, protocol): | ||||
|         logger.debug('Pre connect invoked') | ||||
|         if protocol != 'rdp':  # If connection is not using rdp, skip adding user | ||||
|             return 'ok' | ||||
|         # Well known SSID for Remote Desktop Users | ||||
| @@ -216,38 +227,56 @@ class UDSActorSvc(win32serviceutil.ServiceFramework, CommonService): | ||||
|         ''' | ||||
|         Main service loop | ||||
|         ''' | ||||
|         initCfg() | ||||
|         try: | ||||
|             initCfg() | ||||
|  | ||||
|         logger.debug('running SvcDoRun') | ||||
|         servicemanager.LogMsg(servicemanager.EVENTLOG_INFORMATION_TYPE, | ||||
|                               servicemanager.PYS_SERVICE_STARTED, | ||||
|                               (self._svc_name_, '')) | ||||
|             logger.debug('running SvcDoRun') | ||||
|             servicemanager.LogMsg(servicemanager.EVENTLOG_INFORMATION_TYPE, | ||||
|                                   servicemanager.PYS_SERVICE_STARTED, | ||||
|                                   (self._svc_name_, '')) | ||||
|  | ||||
|         # call the CoInitialize to allow the registration to run in an other | ||||
|         # thread | ||||
|         logger.debug('Initializing com...') | ||||
|         pythoncom.CoInitialize() | ||||
|             # call the CoInitialize to allow the registration to run in an other | ||||
|             # thread | ||||
|             logger.debug('Initializing com...') | ||||
|             pythoncom.CoInitialize() | ||||
|  | ||||
|         # ******************************************************** | ||||
|         # * Ask brokers what to do before proceding to main loop * | ||||
|         # ******************************************************** | ||||
|         if self.interactWithBroker() is False: | ||||
|             logger.debug('Interact with broker returned false, stopping service after a while') | ||||
|             self.notifyStop() | ||||
|             win32event.WaitForSingleObject(self.hWaitStop, 5000) | ||||
|             return | ||||
|             # ******************************************************** | ||||
|             # * Ask brokers what to do before proceding to main loop * | ||||
|             # ******************************************************** | ||||
|             while True: | ||||
|                 brokerConnected = self.interactWithBroker() | ||||
|                 if brokerConnected is False: | ||||
|                     logger.debug('Interact with broker returned false, stopping service after a while') | ||||
|                     self.notifyStop() | ||||
|                     win32event.WaitForSingleObject(self.hWaitStop, 5000) | ||||
|                     return | ||||
|                 elif brokerConnected is True: | ||||
|                     break | ||||
|  | ||||
|         if self.isAlive is False: | ||||
|             logger.debug('The service is not alive after broker interaction, stopping it') | ||||
|             self.notifyStop() | ||||
|             return | ||||
|                 # If brokerConnected returns None, repeat the cycle | ||||
|                 self.doWait(16000)  # Wait for a looong while | ||||
|  | ||||
|         if self.rebootRequested is True: | ||||
|             logger.debug('Reboot has been requested, stopping service') | ||||
|             self.notifyStop() | ||||
|             return | ||||
|             if self.interactWithBroker() is False: | ||||
|                 logger.debug('Interact with broker returned false, stopping service after a while') | ||||
|                 self.notifyStop() | ||||
|                 win32event.WaitForSingleObject(self.hWaitStop, 5000) | ||||
|                 return | ||||
|  | ||||
|         self.initIPC() | ||||
|             if self.isAlive is False: | ||||
|                 logger.debug('The service is not alive after broker interaction, stopping it') | ||||
|                 self.notifyStop() | ||||
|                 return | ||||
|  | ||||
|             if self.rebootRequested is True: | ||||
|                 logger.debug('Reboot has been requested, stopping service') | ||||
|                 self.notifyStop() | ||||
|                 return | ||||
|  | ||||
|             self.initIPC() | ||||
|         except Exception:  # Any init exception wil be caught, service must be then restarted | ||||
|             logger.exception() | ||||
|             logger.debug('Exiting service with failure status') | ||||
|             os._exit(-1)  # pylint: disable=protected-access | ||||
|  | ||||
|         # ******************************** | ||||
|         # * Registers SENS subscriptions * | ||||
| @@ -270,6 +299,17 @@ class UDSActorSvc(win32serviceutil.ServiceFramework, CommonService): | ||||
|  | ||||
|         logger.debug('Registered SENS, running main loop') | ||||
|  | ||||
|         # Execute script in c:\\windows\\post-uds.bat after interacting with broker, if no reboot is requested ofc | ||||
|         # This will be executed only when machine gets "ready" | ||||
|         try: | ||||
|             if os.path.isfile(POST_CMD): | ||||
|                 subprocess.call([POST_CMD, ]) | ||||
|             else: | ||||
|                 logger.info('POST file not found & not executed') | ||||
|         except Exception as e: | ||||
|             # Ignore output of execution command | ||||
|             logger.error('Executing post command give') | ||||
|  | ||||
|         # ********************* | ||||
|         # * Main Service loop * | ||||
|         # ********************* | ||||
| @@ -281,8 +321,12 @@ class UDSActorSvc(win32serviceutil.ServiceFramework, CommonService): | ||||
|             # Process SENS messages, This will be a bit asyncronous (1 second | ||||
|             # delay) | ||||
|             pythoncom.PumpWaitingMessages() | ||||
|             if counter % 10 == 0: | ||||
|                 self.checkIpsChanged() | ||||
|             if counter >= 15:  # Once every 15 seconds | ||||
|                 counter = 0 | ||||
|                 try: | ||||
|                     self.checkIpsChanged() | ||||
|                 except Exception as e: | ||||
|                     logger.error('Error checking ip change: {}'.format(e)) | ||||
|             # In milliseconds, will break | ||||
|             win32event.WaitForSingleObject(self.hWaitStop, 1000) | ||||
|  | ||||
|   | ||||
| @@ -48,7 +48,7 @@ class LocalLogger(object): | ||||
|             filename=os.path.join(tempfile.gettempdir(), 'udsactor.log'), | ||||
|             filemode='a', | ||||
|             format='%(levelname)s %(asctime)s %(message)s', | ||||
|             level=logging.DEBUG | ||||
|             level=logging.INFO | ||||
|         ) | ||||
|         self.logger = logging.getLogger('udsactor') | ||||
|         self.serviceLogger = False | ||||
|   | ||||
| @@ -45,6 +45,7 @@ from udsactor.log import logger | ||||
|  | ||||
|  | ||||
| def getErrorMessage(res=0): | ||||
|     # sys_fs_enc = sys.getfilesystemencoding() or 'mbcs' | ||||
|     msg = win32api.FormatMessage(res) | ||||
|     return msg.decode('windows-1250', 'ignore') | ||||
|  | ||||
| @@ -62,8 +63,9 @@ def getNetworkInfo(): | ||||
|             for ip in obj.IPAddress: | ||||
|                 if ':' in ip:  # Is IPV6, skip this | ||||
|                     continue | ||||
|                 if ip == '' or ip is None: | ||||
|                 if ip is None or ip == '' or ip.startswith('169.254') or ip.startswith('0.'):  # If single link ip, or no ip | ||||
|                     continue | ||||
|                 # logger.debug('Net config found: {}=({}, {})'.format(obj.Caption, obj.MACAddress, ip)) | ||||
|                 yield utils.Bunch(name=obj.Caption, mac=obj.MACAddress, ip=ip) | ||||
|     except Exception: | ||||
|         return | ||||
| @@ -111,7 +113,7 @@ def loggoff(): | ||||
|  | ||||
| def renameComputer(newName): | ||||
|     # Needs admin privileges to work | ||||
|     if ctypes.windll.kernel32.SetComputerNameExW(DWORD(win32con.ComputerNamePhysicalDnsHostname), LPCWSTR(newName)) == 0: | ||||
|     if ctypes.windll.kernel32.SetComputerNameExW(DWORD(win32con.ComputerNamePhysicalDnsHostname), LPCWSTR(newName)) == 0:  # @UndefinedVariable | ||||
|         # win32api.FormatMessage -> returns error string | ||||
|         # win32api.GetLastError -> returns error code | ||||
|         # (just put this comment here to remember to log this when logger is available) | ||||
| @@ -131,6 +133,14 @@ NETSETUP_DEFER_SPN_SET = 0x1000000 | ||||
|  | ||||
|  | ||||
| def joinDomain(domain, ou, account, password, executeInOneStep=False): | ||||
|     ''' | ||||
|     Joins machine to a windows domain | ||||
|     :param domain: Domain to join to | ||||
|     :param ou: Ou that will hold machine | ||||
|     :param account: Account used to join domain | ||||
|     :param password: Password of account used to join domain | ||||
|     :param executeInOneStep: If true, means that this machine has been renamed and wants to add NETSETUP_JOIN_WITH_NEW_NAME to request so we can do rename/join in one step. | ||||
|     ''' | ||||
|     # If account do not have domain, include it | ||||
|     if '@' not in account and '\\' not in account: | ||||
|         if '.' in domain: | ||||
| @@ -138,7 +148,6 @@ def joinDomain(domain, ou, account, password, executeInOneStep=False): | ||||
|         else: | ||||
|             account = domain + '\\' + account | ||||
|  | ||||
|  | ||||
|     # Do log | ||||
|     flags = NETSETUP_ACCT_CREATE | NETSETUP_DOMAIN_JOIN_IF_JOINED | NETSETUP_JOIN_DOMAIN | ||||
|  | ||||
| @@ -162,8 +171,10 @@ def joinDomain(domain, ou, account, password, executeInOneStep=False): | ||||
|     if res != 0: | ||||
|         # Log the error | ||||
|         error = getErrorMessage(res) | ||||
|         print res, error | ||||
|         raise Exception('Error joining domain {}, with credentials {}/*****{}: {}, {}'.format(domain.value, account.value, ', under OU {}'.format(ou.value) if ou.value != None else '', res, error)) | ||||
|         if res == 1355: | ||||
|             error = "DC Is not reachable" | ||||
|         logger.error('Error joining domain: {}, {}'.format(error, res)) | ||||
|         raise Exception('Error joining domain {}, with credentials {}/*****{}: {}, {}'.format(domain, account, ', under OU {}'.format(ou) if ou is not None else '', res, error)) | ||||
|  | ||||
|  | ||||
| def changeUserPassword(user, oldPassword, newPassword): | ||||
| @@ -198,7 +209,7 @@ def getIdleDuration(): | ||||
|     lastInputInfo = LASTINPUTINFO() | ||||
|     lastInputInfo.cbSize = ctypes.sizeof(lastInputInfo) | ||||
|     ctypes.windll.user32.GetLastInputInfo(ctypes.byref(lastInputInfo)) | ||||
|     millis = ctypes.windll.kernel32.GetTickCount() - lastInputInfo.dwTime | ||||
|     millis = ctypes.windll.kernel32.GetTickCount() - lastInputInfo.dwTime  # @UndefinedVariable | ||||
|     return millis / 1000.0 | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -82,13 +82,42 @@ def readConfig(): | ||||
|     except Exception: | ||||
|         return None | ||||
|  | ||||
|  | ||||
| def writeConfig(data): | ||||
| def writeConfig(data, fixPermissions=True): | ||||
|     try: | ||||
|         key = wreg.OpenKey(baseKey, path, 0, wreg.KEY_ALL_ACCESS)  # @UndefinedVariable | ||||
|     except Exception: | ||||
|         key = wreg.CreateKeyEx(baseKey, path, 0, wreg.KEY_ALL_ACCESS)  # @UndefinedVariable | ||||
|         fixRegistryPermissions(key.handle) | ||||
|         if fixPermissions is True: | ||||
|             fixRegistryPermissions(key.handle) | ||||
|  | ||||
|     wreg.SetValueEx(key, "", 0, wreg.REG_BINARY, encoder(cPickle.dumps(data)))  # @UndefinedVariable | ||||
|     wreg.CloseKey(key)  # @UndefinedVariable | ||||
|  | ||||
| def useOldJoinSystem(): | ||||
|     try: | ||||
|         key = wreg.OpenKey(baseKey, 'Software\\UDSEnterpriseActor', 0, wreg.KEY_QUERY_VALUE)  # @UndefinedVariable | ||||
|         try: | ||||
|             data, _ = wreg.QueryValueEx(key, 'join')  # @UndefinedVariable | ||||
|         except Exception: | ||||
|             data = '' | ||||
|         wreg.CloseKey(key)  # @UndefinedVariable | ||||
|     except: | ||||
|         data = '' | ||||
|  | ||||
|     return data == 'old' | ||||
|  | ||||
| # Gives the oportunity to run an application ONE TIME (because, the registry key "run" will be deleted after read) | ||||
| def runApplication(): | ||||
|     try: | ||||
|         key = wreg.OpenKey(baseKey, 'Software\\UDSEnterpriseActor', 0, wreg.KEY_ALL_ACCESS)  # @UndefinedVariable | ||||
|         try: | ||||
|             data, _ = wreg.QueryValueEx(key, 'run')  # @UndefinedVariable | ||||
|             wreg.DeleteValue(key, 'run')  # @UndefinedVariable | ||||
|         except Exception: | ||||
|             data = None | ||||
|         wreg.CloseKey(key)  # @UndefinedVariable | ||||
|     except: | ||||
|         data = None | ||||
|  | ||||
|     return data | ||||
|  | ||||
|   | ||||
							
								
								
									
										15
									
								
								actors/src/update.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
|  | ||||
| function process { | ||||
|     pyuic4 about-dialog.ui -o about_dialog_ui.py -x | ||||
|     pyuic4 message-dialog.ui -o message_dialog_ui.py -x | ||||
|     pyuic4 setup-dialog.ui -o setup_dialog_ui.py -x | ||||
| }     | ||||
|  | ||||
| pyrcc4 -py3 UDSActor.qrc -o UDSActor_rc.py | ||||
|  | ||||
|  | ||||
| # process current directory ui's | ||||
| process | ||||
|  | ||||
							
								
								
									
										4
									
								
								client/full/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | ||||
| /bin | ||||
| /udsclient_* | ||||
| /udsclient-*.tar.gz | ||||
| /*.rpm | ||||
| @@ -1,6 +1,6 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <projectDescription> | ||||
| 	<name>linuxActor</name> | ||||
| 	<name>UDSclient</name> | ||||
| 	<comment></comment> | ||||
| 	<projects> | ||||
| 	</projects> | ||||
| @@ -1,8 +1,8 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <?eclipse-pydev version="1.0"?><pydev_project> | ||||
| <pydev_property name="org.python.pydev.PYTHON_PROJECT_INTERPRETER">Default</pydev_property> | ||||
| <pydev_property name="org.python.pydev.PYTHON_PROJECT_VERSION">python 2.7</pydev_property> | ||||
| <pydev_pathproperty name="org.python.pydev.PROJECT_SOURCE_PATH"> | ||||
| <path>/${PROJECT_DIR_NAME}/src</path> | ||||
| </pydev_pathproperty> | ||||
| <pydev_property name="org.python.pydev.PYTHON_PROJECT_VERSION">python 2.7</pydev_property> | ||||
| <pydev_property name="org.python.pydev.PYTHON_PROJECT_INTERPRETER">Default</pydev_property> | ||||
| </pydev_project> | ||||
							
								
								
									
										4
									
								
								client/full/linux/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | ||||
| /udsclient-opensuse-[0-9]*.spec | ||||
| /udsclient-[0-9]*.spec | ||||
| /debian/udsclient | ||||
| /targz | ||||
							
								
								
									
										51
									
								
								client/full/linux/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,51 @@ | ||||
| #!/usr/bin/make -f | ||||
| # -*- makefile -*- | ||||
|  | ||||
| # Version | ||||
| # VERSION := 1.7.5 | ||||
|  | ||||
| # Directories | ||||
| SOURCEDIR := ../src | ||||
| LIBDIR := $(DESTDIR)/usr/lib/UDSClient | ||||
| BINDIR := $(DESTDIR)/usr/bin | ||||
| SBINDIR = $(DESTDIR)/usr/sbin | ||||
| APPSDIR := $(DESTDIR)/usr/share/applications | ||||
|  | ||||
| PYC := $(shell find $(SOURCEDIR) -name '*.py[co]') | ||||
| CACHES := $(shell find $(SOURCEDIR) -name '__pycache__') | ||||
|  | ||||
| clean: | ||||
| 	rm -rf $(PYC) $(CACHES) $(DESTDIR) | ||||
| install: | ||||
| 	rm -rf $(DESTDIR) | ||||
| 	mkdir -p $(LIBDIR) | ||||
| 	#mkdir -p $(BINDIR) | ||||
| 	#mkdir -p $(SBINDIR) | ||||
| 	mkdir -p $(APPSDIR) | ||||
| 	 | ||||
| 	mkdir $(LIBDIR)/uds | ||||
| 	 | ||||
| 	# Cleans up .pyc and cache folders | ||||
| 	rm -f $(PYC) $(CACHES) | ||||
| 	 | ||||
| 	cp $(SOURCEDIR)/uds/*.py $(LIBDIR)/uds | ||||
|  | ||||
| 	cp $(SOURCEDIR)/UDS*.py $(LIBDIR) | ||||
| 	 | ||||
| 	 | ||||
| 	# URL Catchers elements for gnome/kde | ||||
| 	cp desktop/UDSClient.desktop $(APPSDIR) | ||||
| 	 | ||||
| 	chmod 755 $(LIBDIR)/UDSClient.py | ||||
|  | ||||
| ifeq ($(DISTRO),targz) | ||||
| 	cp installer.sh $(DESTDIR)/install.sh | ||||
| 	tar czvf ../udsclient-$(VERSION).tar.gz -C $(DESTDIR) . | ||||
| endif | ||||
|  | ||||
| 	 | ||||
| 	# chmod 0755 $(BINDIR)/udsclient | ||||
| uninstall: | ||||
| 	rm -rf $(LIBDIR) | ||||
| 	# rm -f $(BINDIR)/udsclient | ||||
| 	#  rm -rf $(CFGDIR) | ||||
							
								
								
									
										36
									
								
								client/full/linux/build-packages.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,36 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
| VERSION=`cat ../../../VERSION` | ||||
| RELEASE=1 | ||||
| # Debian based | ||||
| dpkg-buildpackage -b | ||||
|  | ||||
| # Now rpm based | ||||
| top=`pwd` | ||||
|  | ||||
| cat udsclient-template.spec |  | ||||
|   sed -e s/"version 0.0.0"/"version ${VERSION}"/g | | ||||
|   sed -e s/"release 1"/"release ${RELEASE}"/g > udsclient-$VERSION.spec | ||||
|    | ||||
| # Now fix dependencies for opensuse | ||||
| cat udsclient-template.spec |  | ||||
|   sed -e s/"version 0.0.0"/"version ${VERSION}"/g | | ||||
|   sed -e s/"name udsclient"/"name udsclient-opensuse"/g | | ||||
|   sed -e s/"PyQt4"/"python-qt4"/g | | ||||
|   sed -e s/"libXScrnSaver"/"libXss1"/g > udsclient-opensuse-$VERSION.spec | ||||
|  | ||||
|  | ||||
| # Right now, udsactor-xrdp-.spec is not needed | ||||
| for pkg in udsclient-$VERSION.spec udsclient-opensuse-$VERSION.spec; do | ||||
|      | ||||
|     rm -rf rpm | ||||
|     for folder in SOURCES BUILD RPMS SPECS SRPMS; do | ||||
|         mkdir -p rpm/$folder | ||||
|     done | ||||
|      | ||||
|     rpmbuild -v -bb --clean --buildroot=$top/rpm/BUILD/$pkg-root --target noarch $pkg 2>&1 | ||||
| done | ||||
|  | ||||
| #rm udsclient-$VERSION | ||||
|  | ||||
| make DESTDIR=targz DISTRO=targz VERSION=${VERSION} install | ||||
							
								
								
									
										3
									
								
								client/full/linux/debian/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| /udsactor/ | ||||
| /udsactor-xrdp/ | ||||
| /udsactor-nx/ | ||||
							
								
								
									
										29
									
								
								client/full/linux/debian/changelog
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,29 @@ | ||||
| udsclient (2.1.0) stable; urgency=medium | ||||
|  | ||||
|   * Updated release | ||||
|  | ||||
|  -- Adolfo Gómez García <agomez@virtualcable.es>  Sun, 23 Oct 2016 21:12:23 +0200 | ||||
|  | ||||
| udsclient (2.0.0) stable; urgency=medium | ||||
|  | ||||
|   * Release upgrade | ||||
|  | ||||
|  -- Adolfo Gómez García <agomez@virtualcable.es>  Tue, 01 Mar 2016 09:33:18 +0100 | ||||
|  | ||||
| udsclient (1.9.1) stable; urgency=medium | ||||
|  | ||||
|   * Minor fixes & making version match UDS version | ||||
|  | ||||
|  -- Adolfo Gómez García <agomez@virtualcable.es>  Tue, 01 Mar 2016 03:02:37 +0100 | ||||
|  | ||||
| udsclient (1.9.0) stable; urgency=medium | ||||
|  | ||||
|   * Minor fixes & making version match UDS version | ||||
|  | ||||
|  -- Adolfo Gómez García <agomez@virtualcable.es>  Tue, 05 May 2015 07:03:47 +0200 | ||||
|  | ||||
| udsclient (1.7.5) stable; urgency=medium | ||||
|  | ||||
|   * Initial release. | ||||
|  | ||||
|  -- Adolfo Gómez García <agomez@virtualcable.es>  Fri, 10 Apr 2015 05:32:41 +0100 | ||||
							
								
								
									
										1
									
								
								client/full/linux/debian/compat
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| 9 | ||||
							
								
								
									
										15
									
								
								client/full/linux/debian/control
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| Source: udsclient | ||||
| Section: admin | ||||
| Priority: optional | ||||
| Maintainer: Adolfo Gómez García <agomez@virtualcable.es> | ||||
| Build-Depends: debhelper (>= 7), po-debconf | ||||
| Standards-Version: 3.9.2 | ||||
| Homepage: http://www.virtualcable.es | ||||
|  | ||||
| Package: udsclient | ||||
| Section: admin | ||||
| Priority: optional | ||||
| Architecture: all | ||||
| Depends: python-paramiko (>=0.8.2), python-qt4 (>=4.9), python-six(>=1.1), python (>=2.7), rdesktop | freerdp-x11, desktop-file-utils, ${misc:Depends} | ||||
| Description: Client connector for Universal Desktop Services (UDS) Broker | ||||
|  This package provides the required components to allow this machine to connect to services provided by UDS Broker. | ||||
| @@ -1,5 +1,5 @@ | ||||
| Format-Specification: http://svn.debian.org/wsvn/dep/web/deps/dep5.mdwn?op=file&rev=135 | ||||
| Name: udsactor | ||||
| Name: udsclient | ||||
| Maintainer: Adolfo Gómez García | ||||
| Source: http://www.udsenterprise.com/ | ||||
| 
 | ||||
| @@ -23,4 +23,4 @@ with this program; if not, write to the Free Software Foundation, Inc., | ||||
| . | ||||
| On Debian systems, the full text of the GNU General Public | ||||
| License version 2 can be found in the file | ||||
| `/usr/share/common-licenses/GPL-2'. | ||||
| `/usr/share/common-licenses/GPL-2'. | ||||
							
								
								
									
										2
									
								
								client/full/linux/debian/files
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,2 @@ | ||||
| udsclient_2.1.0_all.deb admin optional | ||||
| udsclient_2.1.0_amd64.buildinfo admin optional | ||||
| @@ -21,7 +21,7 @@ install: build | ||||
| 	dh_testroot | ||||
| 	dh_prep | ||||
| 	dh_installdirs | ||||
| 	$(MAKE) DESTDIR=$(CURDIR)/debian/udsactor install | ||||
| 	$(MAKE) DESTDIR=$(CURDIR)/debian/udsclient install | ||||
| binary-arch: build install | ||||
| 	# emptyness | ||||
| binary-indep: build install | ||||
							
								
								
									
										17
									
								
								client/full/linux/debian/udsclient.log
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,17 @@ | ||||
| dh_prep | ||||
| dh_installdirs | ||||
| dh_installchangelogs | ||||
| dh_installdocs | ||||
| dh_installdebconf | ||||
| dh_installinit | ||||
| dh_compress | ||||
| dh_link | ||||
| dh_fixperms | ||||
| dh_installdeb | ||||
| dh_shlibdeps | ||||
| dh_gencontrol | ||||
| dh_md5sums | ||||
| dh_builddeb | ||||
| dh_builddeb | ||||
| dh_builddeb | ||||
| dh_builddeb | ||||
							
								
								
									
										21
									
								
								client/full/linux/debian/udsclient.postinst
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| . /usr/share/debconf/confmodule | ||||
|  | ||||
| set -e | ||||
| case "$1" in | ||||
| 	configure) | ||||
|     ;; | ||||
|  | ||||
|     abort-upgrade|abort-remove|abort-deconfigure) | ||||
|     ;; | ||||
|  | ||||
|     *) | ||||
|         echo "postinst called with unknown argument \`$1'" >&2 | ||||
|         exit 1 | ||||
|     ;; | ||||
| esac | ||||
|  | ||||
| #DEBHELPER# | ||||
|  | ||||
| exit 0 | ||||
							
								
								
									
										6
									
								
								client/full/linux/debian/udsclient.postrm
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,6 @@ | ||||
| #!/bin/sh -e | ||||
|  | ||||
| . /usr/share/debconf/confmodule | ||||
|  | ||||
| set -e | ||||
|  | ||||
							
								
								
									
										1
									
								
								client/full/linux/debian/udsclient.prerm
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| #! /bin/bash -e | ||||
| @@ -1,19 +1,20 @@ | ||||
| Template: udsactor/server | ||||
| Template: udsactor/host | ||||
| Type: string | ||||
| Default: | ||||
| _Description: UDS Server address: | ||||
| Description: UDS Server address: | ||||
|  The actor needs the address of the server in order to communicate with it. | ||||
|  Provide here full address (or i) of the UDS server | ||||
| 
 | ||||
| Template: udsactor/secure | ||||
| Type: boolean | ||||
| Default: true | ||||
| _Description: Use secure (https) connection to communicate with UDS server? | ||||
| Description: Use secure (https) connection to communicate with UDS server? | ||||
|  If selected, the communication will be done using https.  | ||||
|  If not selected, the communication will be done using http | ||||
| 
 | ||||
| Template: udsactor/timeout | ||||
| Template: udsactor/masterKey | ||||
| Type: string | ||||
| Default: 5 | ||||
| _Description: Timeout in communications with UDS server: | ||||
|  The timeout is expressed in seconds  | ||||
| Default: | ||||
| Description: Master Key: | ||||
|  This key is available on UDS Administration interface. | ||||
|  Look for it under configuration, on Security tab. | ||||
							
								
								
									
										11
									
								
								client/full/linux/desktop/UDSClient.desktop
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | ||||
| [Desktop Entry] | ||||
| Name=UDSClient | ||||
| Comment=UDS Helper | ||||
| Keywords=uds;client;vdi; | ||||
| Exec=/usr/lib/UDSClient/UDSClient.py %u | ||||
| Icon=help-browser | ||||
| StartupNotify=true | ||||
| Terminal=false | ||||
| Type=Application | ||||
| Categories=Utility; | ||||
| MimeType=x-scheme-handler/uds;x-scheme-handler/udss; | ||||
							
								
								
									
										14
									
								
								client/full/linux/installer.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| cp -r usr/lib/UDSClient /usr/lib/UDSClient | ||||
| cp -r usr/share/applications /usr/lib/applications -R | ||||
| update-desktop-database | ||||
|  | ||||
| echo "Installation process done." | ||||
| echo "Remembar that the following packages must be installed on system:" | ||||
| echo "* Python paramiko" | ||||
| echo "* Python pyqt4" | ||||
| echo "Theese packages (as their names), are dependent on your platform, so you must locate and install them" | ||||
| echo "You can install them directly on any platform with pip, using this simple command: " | ||||
| echo "pip install PyQt4 paramiko" | ||||
|  | ||||
							
								
								
									
										3
									
								
								client/full/linux/readme.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| UDSClient is the client connector needed to get acccess to services managed by UDS Broker. | ||||
|  | ||||
| Please, visit http://www.udsenterprise.com for more information | ||||
							
								
								
									
										51
									
								
								client/full/linux/udsclient-template.spec
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,51 @@ | ||||
| %define _topdir %(echo $PWD)/rpm | ||||
| %define name udsclient | ||||
| %define version 0.0.0 | ||||
| %define release 1 | ||||
| %define buildroot %{_topdir}/%{name}-%{version}-%{release}-root | ||||
|  | ||||
| BuildRoot: %{buildroot}  | ||||
| Name: %{name} | ||||
| Version: %{version} | ||||
| Release: %{release} | ||||
| Summary: Client for Universal Desktop Services (UDS) Broker | ||||
| License: BSD3 | ||||
| Group: Applications/Productivity | ||||
| Requires: python-six python-paramiko PyQt4 | ||||
| Vendor: Virtual Cable S.L.U. | ||||
| URL: http://www.udsenterprise.com | ||||
| Provides: udsclient | ||||
|  | ||||
| %define _rpmdir ../ | ||||
| %define _rpmfilename %%{NAME}-%%{VERSION}-%%{RELEASE}.%%{ARCH}.rpm | ||||
|  | ||||
|  | ||||
| %install | ||||
| curdir=`pwd` | ||||
| cd ../.. | ||||
| make DESTDIR=$RPM_BUILD_ROOT DISTRO=rh install | ||||
| cd $curdir | ||||
|  | ||||
| %post | ||||
| /usr/bin/update-desktop-database | ||||
|  | ||||
| %clean | ||||
| rm -rf $RPM_BUILD_ROOT | ||||
| curdir=`pwd` | ||||
| cd ../.. | ||||
| make DESTDIR=$RPM_BUILD_ROOT DISTRO=rh clean | ||||
| cd $curdir | ||||
|  | ||||
|  | ||||
| %postun | ||||
| # And, posibly, the .pyc leaved behind on /usr/share/UDSActor | ||||
| rm -rf /usr/share/UDClient > /dev/null 2>&1 | ||||
| /usr/bin/update-desktop-database | ||||
|  | ||||
| %description | ||||
| This package provides the required components to allow connection to services offered by UDS Broker. | ||||
|  | ||||
| %files | ||||
| %defattr(-,root,root) | ||||
| /usr/lib/UDSClient/* | ||||
| /usr/share/applications/UDSClient.desktop | ||||
							
								
								
									
										4
									
								
								client/full/src/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | ||||
| /build | ||||
| /dist | ||||
| UDSClient.dmg | ||||
| UDSClient.pkg | ||||
							
								
								
									
										310
									
								
								client/full/src/UDSClient.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,310 @@ | ||||
| #!/usr/bin/env python | ||||
| # -*- coding: utf-8 -*- | ||||
| # | ||||
| # Copyright (c) 2017 Virtual Cable S.L. | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without modification, | ||||
| # are permitted provided that the following conditions are met: | ||||
| # | ||||
| #    * Redistributions of source code must retain the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer. | ||||
| #    * Redistributions in binary form must reproduce the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer in the documentation | ||||
| #      and/or other materials provided with the distribution. | ||||
| #    * Neither the name of Virtual Cable S.L. nor the names of its contributors | ||||
| #      may be used to endorse or promote products derived from this software | ||||
| #      without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||
| # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | ||||
| # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | ||||
| # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | ||||
| # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
| # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
|  | ||||
| ''' | ||||
| @author: Adolfo Gómez, dkmaster at dkmon dot com | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import sys | ||||
| from PyQt4 import QtCore, QtGui | ||||
| import six | ||||
|  | ||||
| from uds.rest import RestRequest | ||||
| from uds.forward import forward | ||||
| from uds.log import logger | ||||
| from uds import tools | ||||
| from uds import VERSION | ||||
|  | ||||
| import webbrowser | ||||
|  | ||||
| from UDSWindow import Ui_MainWindow | ||||
|  | ||||
| class RetryException(Exception): | ||||
|     pass | ||||
|  | ||||
| class UDSClient(QtGui.QMainWindow): | ||||
|  | ||||
|     ticket = None | ||||
|     scrambler = None | ||||
|     withError = False | ||||
|     animTimer = None | ||||
|     anim = 0 | ||||
|     animInverted = False | ||||
|  | ||||
|     def __init__(self): | ||||
|         QtGui.QMainWindow.__init__(self) | ||||
|         self.setWindowFlags(QtCore.Qt.FramelessWindowHint | QtCore.Qt.WindowStaysOnTopHint) | ||||
|  | ||||
|         self.ui = Ui_MainWindow() | ||||
|         self.ui.setupUi(self) | ||||
|  | ||||
|         self.ui.progressBar.setValue(0) | ||||
|         self.ui.cancelButton.clicked.connect(self.cancelPushed) | ||||
|  | ||||
|         self.ui.info.setText('Initializing...') | ||||
|  | ||||
|         screen = QtGui.QDesktopWidget().screenGeometry() | ||||
|         mysize = self.geometry() | ||||
|         hpos = (screen.width() - mysize.width()) / 2 | ||||
|         vpos = (screen.height() - mysize.height() - mysize.height()) / 2 | ||||
|         self.move(hpos, vpos) | ||||
|  | ||||
|         self.animTimer = QtCore.QTimer() | ||||
|         QtCore.QObject.connect(self.animTimer, QtCore.SIGNAL('timeout()'), self.updateAnim) | ||||
|  | ||||
|         self.activateWindow() | ||||
|  | ||||
|         self.startAnim() | ||||
|  | ||||
|  | ||||
|     def closeWindow(self): | ||||
|         self.close() | ||||
|  | ||||
|     def processError(self, data): | ||||
|         if 'error' in data: | ||||
|             # QtGui.QMessageBox.critical(self, 'Request error {}'.format(data.get('retryable', '0')), data['error'], QtGui.QMessageBox.Ok) | ||||
|             if data.get('retryable', '0') == '1': | ||||
|                 raise RetryException(data['error']) | ||||
|  | ||||
|             raise Exception(data['error']) | ||||
|             # QtGui.QMessageBox.critical(self, 'Request error', rest.data['error'], QtGui.QMessageBox.Ok) | ||||
|             # self.closeWindow() | ||||
|             # return | ||||
|  | ||||
|     def showError(self, e): | ||||
|         logger.error('got error: {}'.format(e)) | ||||
|         self.stopAnim() | ||||
|         self.ui.info.setText('UDS Plugin Error')  # In fact, main window is hidden, so this is not visible... :) | ||||
|         self.closeWindow() | ||||
|         QtGui.QMessageBox.critical(None, 'UDS Plugin Error', '{}'.format(e), QtGui.QMessageBox.Ok) | ||||
|         self.withError = True | ||||
|  | ||||
|     def cancelPushed(self): | ||||
|         self.close() | ||||
|  | ||||
|     @QtCore.pyqtSlot() | ||||
|     def updateAnim(self): | ||||
|         self.anim += 2 | ||||
|         if self.anim > 99: | ||||
|             self.animInverted = not self.animInverted | ||||
|             self.ui.progressBar.setInvertedAppearance(self.animInverted) | ||||
|             self.anim = 0 | ||||
|  | ||||
|         self.ui.progressBar.setValue(self.anim) | ||||
|  | ||||
|     def startAnim(self): | ||||
|         self.ui.progressBar.invertedAppearance = False | ||||
|         self.anim = 0 | ||||
|         self.animInverted = False | ||||
|         self.ui.progressBar.setInvertedAppearance(self.animInverted) | ||||
|         self.animTimer.start(40) | ||||
|  | ||||
|     def stopAnim(self): | ||||
|         self.ui.progressBar.invertedAppearance = False | ||||
|         self.animTimer.stop() | ||||
|  | ||||
|     @QtCore.pyqtSlot() | ||||
|     def getVersion(self): | ||||
|         self.req = RestRequest('', self, self.version) | ||||
|         self.req.get() | ||||
|  | ||||
|     @QtCore.pyqtSlot(dict) | ||||
|     def version(self, data): | ||||
|         try: | ||||
|             self.processError(data) | ||||
|             self.ui.info.setText('Processing...') | ||||
|  | ||||
|             if data['result']['requiredVersion'] > VERSION: | ||||
|                 QtGui.QMessageBox.critical(self, 'Upgrade required', 'A newer connector version is required.\nA browser will be opened to download it.', QtGui.QMessageBox.Ok) | ||||
|                 webbrowser.open(data['result']['downloadUrl']) | ||||
|                 self.closeWindow() | ||||
|                 return | ||||
|             self.getTransportData() | ||||
|  | ||||
|         except RetryException as e: | ||||
|             self.ui.info.setText(six.text_type(e)) | ||||
|             QtCore.QTimer.singleShot(1000, self.getVersion) | ||||
|  | ||||
|         except Exception as e: | ||||
|             self.showError(e) | ||||
|  | ||||
|  | ||||
|     @QtCore.pyqtSlot() | ||||
|     def getTransportData(self): | ||||
|         try: | ||||
|             self.req = RestRequest('/{}/{}'.format(self.ticket, self.scrambler), self, self.transportDataReceived, params={'hostname': tools.getHostName(), 'version': VERSION}) | ||||
|             self.req.get() | ||||
|         except Exception as e: | ||||
|             logger.exception('Got exception: {}'.format(e)) | ||||
|             raise e | ||||
|  | ||||
|  | ||||
|     @QtCore.pyqtSlot(dict) | ||||
|     def transportDataReceived(self, data): | ||||
|         logger.debug('Transport data received') | ||||
|         try: | ||||
|             self.processError(data) | ||||
|  | ||||
|             script = data['result'].decode('base64').decode('bz2') | ||||
|  | ||||
|             self.stopAnim() | ||||
|  | ||||
|             if 'darwin' in sys.platform: | ||||
|                 self.showMinimized() | ||||
|  | ||||
|             QtCore.QTimer.singleShot(3000, self.endScript) | ||||
|             self.hide() | ||||
|  | ||||
|             six.exec_(script, globals(), {'parent': self}) | ||||
|  | ||||
|         except RetryException as e: | ||||
|             self.ui.info.setText(six.text_type(e) + ', retrying access...') | ||||
|             # Retry operation in ten seconds | ||||
|             QtCore.QTimer.singleShot(10000, self.getTransportData) | ||||
|  | ||||
|         except Exception as e: | ||||
|             logger.exception('Got exception executing script:') | ||||
|             self.showError(e) | ||||
|  | ||||
|     def endScript(self): | ||||
|         # After running script, wait for stuff | ||||
|         try: | ||||
|             tools.waitForTasks() | ||||
|         except Exception: | ||||
|             pass | ||||
|  | ||||
|         try: | ||||
|             tools.unlinkFiles() | ||||
|         except Exception: | ||||
|             pass | ||||
|  | ||||
|         try: | ||||
|             tools.execBeforeExit() | ||||
|         except Exception: | ||||
|             pass | ||||
|  | ||||
|         self.closeWindow() | ||||
|  | ||||
|     def start(self): | ||||
|         ''' | ||||
|         Starts proccess by requesting version info | ||||
|         ''' | ||||
|         self.ui.info.setText('Initializing...') | ||||
|         QtCore.QTimer.singleShot(100, self.getVersion) | ||||
|  | ||||
|  | ||||
| def done(data): | ||||
|     QtGui.QMessageBox.critical(None, 'Notice', six.text_type(data.data), QtGui.QMessageBox.Ok) | ||||
|     sys.exit(0) | ||||
|  | ||||
| # Ask user to aprobe endpoint | ||||
| def approveHost(host, parentWindow=None): | ||||
|     settings = QtCore.QSettings() | ||||
|     settings.beginGroup('endpoints') | ||||
|  | ||||
|     approved = settings.value(host, False).toBool() | ||||
|  | ||||
|     errorString = '<p>The server <b>{}</b> must be approved:</p>'.format(host) | ||||
|     errorString += '<p>Only approve UDS servers that you trust to avoid security issues.</p>' | ||||
|  | ||||
|     if approved or QtGui.QMessageBox.warning(parentWindow, 'ACCESS Warning', errorString, QtGui.QMessageBox.Yes | QtGui.QMessageBox.No) == QtGui.QMessageBox.Yes: | ||||
|         settings.setValue(host, True) | ||||
|         approved = True | ||||
|  | ||||
|     settings.endGroup() | ||||
|     return approved | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     logger.debug('Initializing connector') | ||||
|     # Initialize app | ||||
|     app = QtGui.QApplication(sys.argv) | ||||
|  | ||||
|     # Set several info for settings | ||||
|     QtCore.QCoreApplication.setOrganizationName('Virtual Cable S.L.U.') | ||||
|     QtCore.QCoreApplication.setApplicationName('UDS Connector') | ||||
|  | ||||
|     if 'darwin' not in sys.platform: | ||||
|         logger.debug('Mac OS *NOT* Detected') | ||||
|         app.setStyle('plastique') | ||||
|  | ||||
|     if six.PY3 is False: | ||||
|         logger.debug('Fixing threaded execution of commands') | ||||
|         import threading | ||||
|         threading._DummyThread._Thread__stop = lambda x: 42 | ||||
|  | ||||
|     # First parameter must be url | ||||
|     try: | ||||
|         uri = sys.argv[1] | ||||
|  | ||||
|         if uri == '--test': | ||||
|             sys.exit(0) | ||||
|  | ||||
|         logger.debug('URI: {}'.format(uri)) | ||||
|         if uri[:6] != 'uds://' and uri[:7] != 'udss://': | ||||
|             raise Exception() | ||||
|  | ||||
|         ssl = uri[3] == 's' | ||||
|         host, UDSClient.ticket, UDSClient.scrambler = uri.split('//')[1].split('/') | ||||
|         logger.debug('ssl: {}, host:{}, ticket:{}, scrambler:{}'.format(ssl, host, UDSClient.ticket, UDSClient.scrambler)) | ||||
|  | ||||
|     except Exception: | ||||
|         logger.debug('Detected execution without valid URI, exiting') | ||||
|         QtGui.QMessageBox.critical(None, 'Notice', 'UDS Client Version {}'.format(VERSION), QtGui.QMessageBox.Ok) | ||||
|         sys.exit(1) | ||||
|  | ||||
|     # Setup REST api endpoint | ||||
|     RestRequest.restApiUrl = '{}://{}/rest/client'.format(['http', 'https'][ssl], host) | ||||
|     logger.debug('Setting request URL to {}'.format(RestRequest.restApiUrl)) | ||||
|     # RestRequest.restApiUrl = 'https://172.27.0.1/rest/client' | ||||
|  | ||||
|     try: | ||||
|         logger.debug('Starting execution') | ||||
|  | ||||
|         # Approbe before going on | ||||
|         if approveHost(host) is False: | ||||
|             raise Exception('Host {} was not approved'.format(host)) | ||||
|  | ||||
|         win = UDSClient() | ||||
|         win.show() | ||||
|  | ||||
|  | ||||
|         win.start() | ||||
|  | ||||
|         exitVal = app.exec_() | ||||
|         logger.debug('Execution finished correctly') | ||||
|  | ||||
|     except Exception as e: | ||||
|         logger.exception('Got an exception executing client:') | ||||
|         exitVal = 128 | ||||
|         QtGui.QMessageBox.critical(None, 'Error', six.text_type(e), QtGui.QMessageBox.Ok) | ||||
|  | ||||
|     logger.debug('Exiting') | ||||
|     sys.exit(exitVal) | ||||
|  | ||||
							
								
								
									
										6
									
								
								client/full/src/UDSResources.qrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,6 @@ | ||||
| <RCC> | ||||
|   <qresource prefix="images"> | ||||
|     <file alias="logo-uds-small">images/logo-uds-small.png</file> | ||||
|     <file alias="logo-uds-big">images/logo-uds.png</file> | ||||
|   </qresource> | ||||
| </RCC> | ||||
							
								
								
									
										2471
									
								
								client/full/src/UDSResources_rc.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										105
									
								
								client/full/src/UDSWindow.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,105 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # Form implementation generated from reading ui file 'UDSWindow.ui' | ||||
| # | ||||
| # Created: Mon Apr 27 21:41:43 2015 | ||||
| #      by: PyQt4 UI code generator 4.11.2 | ||||
| # | ||||
| # WARNING! All changes made in this file will be lost! | ||||
|  | ||||
| from PyQt4 import QtCore, QtGui | ||||
|  | ||||
| try: | ||||
|     _fromUtf8 = QtCore.QString.fromUtf8 | ||||
| except AttributeError: | ||||
|     def _fromUtf8(s): | ||||
|         return s | ||||
|  | ||||
| try: | ||||
|     _encoding = QtGui.QApplication.UnicodeUTF8 | ||||
|     def _translate(context, text, disambig): | ||||
|         return QtGui.QApplication.translate(context, text, disambig, _encoding) | ||||
| except AttributeError: | ||||
|     def _translate(context, text, disambig): | ||||
|         return QtGui.QApplication.translate(context, text, disambig) | ||||
|  | ||||
| class Ui_MainWindow(object): | ||||
|     def setupUi(self, MainWindow): | ||||
|         MainWindow.setObjectName(_fromUtf8("MainWindow")) | ||||
|         MainWindow.setWindowModality(QtCore.Qt.NonModal) | ||||
|         MainWindow.resize(259, 185) | ||||
|         MainWindow.setCursor(QtGui.QCursor(QtCore.Qt.BusyCursor)) | ||||
|         icon = QtGui.QIcon() | ||||
|         icon.addPixmap(QtGui.QPixmap(_fromUtf8(":/images/logo-uds-small")), QtGui.QIcon.Normal, QtGui.QIcon.Off) | ||||
|         MainWindow.setWindowIcon(icon) | ||||
|         MainWindow.setWindowOpacity(1.0) | ||||
|         self.centralwidget = QtGui.QWidget(MainWindow) | ||||
|         self.centralwidget.setAutoFillBackground(True) | ||||
|         self.centralwidget.setObjectName(_fromUtf8("centralwidget")) | ||||
|         self.verticalLayout_2 = QtGui.QVBoxLayout(self.centralwidget) | ||||
|         self.verticalLayout_2.setSpacing(4) | ||||
|         self.verticalLayout_2.setMargin(4) | ||||
|         self.verticalLayout_2.setObjectName(_fromUtf8("verticalLayout_2")) | ||||
|         self.frame = QtGui.QFrame(self.centralwidget) | ||||
|         self.frame.setFrameShape(QtGui.QFrame.StyledPanel) | ||||
|         self.frame.setFrameShadow(QtGui.QFrame.Raised) | ||||
|         self.frame.setObjectName(_fromUtf8("frame")) | ||||
|         self.verticalLayout_3 = QtGui.QVBoxLayout(self.frame) | ||||
|         self.verticalLayout_3.setSpacing(4) | ||||
|         self.verticalLayout_3.setMargin(4) | ||||
|         self.verticalLayout_3.setObjectName(_fromUtf8("verticalLayout_3")) | ||||
|         self.verticalLayout = QtGui.QVBoxLayout() | ||||
|         self.verticalLayout.setObjectName(_fromUtf8("verticalLayout")) | ||||
|         self.image = QtGui.QLabel(self.frame) | ||||
|         self.image.setMinimumSize(QtCore.QSize(0, 24)) | ||||
|         self.image.setAutoFillBackground(True) | ||||
|         self.image.setText(_fromUtf8("")) | ||||
|         self.image.setPixmap(QtGui.QPixmap(_fromUtf8(":/images/logo-uds-small"))) | ||||
|         self.image.setScaledContents(False) | ||||
|         self.image.setAlignment(QtCore.Qt.AlignCenter) | ||||
|         self.image.setObjectName(_fromUtf8("image")) | ||||
|         self.verticalLayout.addWidget(self.image) | ||||
|         self.info = QtGui.QLabel(self.frame) | ||||
|         self.info.setMaximumSize(QtCore.QSize(16777215, 16)) | ||||
|         self.info.setObjectName(_fromUtf8("info")) | ||||
|         self.verticalLayout.addWidget(self.info) | ||||
|         self.progressBar = QtGui.QProgressBar(self.frame) | ||||
|         self.progressBar.setProperty("value", 24) | ||||
|         self.progressBar.setTextVisible(False) | ||||
|         self.progressBar.setObjectName(_fromUtf8("progressBar")) | ||||
|         self.verticalLayout.addWidget(self.progressBar) | ||||
|         self.horizontalLayout = QtGui.QHBoxLayout() | ||||
|         self.horizontalLayout.setObjectName(_fromUtf8("horizontalLayout")) | ||||
|         spacerItem = QtGui.QSpacerItem(40, 20, QtGui.QSizePolicy.Expanding, QtGui.QSizePolicy.Minimum) | ||||
|         self.horizontalLayout.addItem(spacerItem) | ||||
|         self.cancelButton = QtGui.QPushButton(self.frame) | ||||
|         self.cancelButton.setDefault(True) | ||||
|         self.cancelButton.setFlat(False) | ||||
|         self.cancelButton.setObjectName(_fromUtf8("cancelButton")) | ||||
|         self.horizontalLayout.addWidget(self.cancelButton) | ||||
|         spacerItem1 = QtGui.QSpacerItem(40, 20, QtGui.QSizePolicy.Expanding, QtGui.QSizePolicy.Minimum) | ||||
|         self.horizontalLayout.addItem(spacerItem1) | ||||
|         self.verticalLayout.addLayout(self.horizontalLayout) | ||||
|         self.verticalLayout_3.addLayout(self.verticalLayout) | ||||
|         self.verticalLayout_2.addWidget(self.frame) | ||||
|         MainWindow.setCentralWidget(self.centralwidget) | ||||
|  | ||||
|         self.retranslateUi(MainWindow) | ||||
|         QtCore.QMetaObject.connectSlotsByName(MainWindow) | ||||
|  | ||||
|     def retranslateUi(self, MainWindow): | ||||
|         MainWindow.setWindowTitle(_translate("MainWindow", "UDS Connection", None)) | ||||
|         self.info.setText(_translate("MainWindow", "TextLabel", None)) | ||||
|         self.cancelButton.setText(_translate("MainWindow", "Cancel", None)) | ||||
|  | ||||
| import UDSResources_rc | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     import sys | ||||
|     app = QtGui.QApplication(sys.argv) | ||||
|     MainWindow = QtGui.QMainWindow() | ||||
|     ui = Ui_MainWindow() | ||||
|     ui.setupUi(MainWindow) | ||||
|     MainWindow.show() | ||||
|     sys.exit(app.exec_()) | ||||
|  | ||||
							
								
								
									
										160
									
								
								client/full/src/UDSWindow.ui
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,160 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <ui version="4.0"> | ||||
|  <class>MainWindow</class> | ||||
|  <widget class="QMainWindow" name="MainWindow"> | ||||
|   <property name="windowModality"> | ||||
|    <enum>Qt::NonModal</enum> | ||||
|   </property> | ||||
|   <property name="geometry"> | ||||
|    <rect> | ||||
|     <x>0</x> | ||||
|     <y>0</y> | ||||
|     <width>259</width> | ||||
|     <height>185</height> | ||||
|    </rect> | ||||
|   </property> | ||||
|   <property name="cursor"> | ||||
|    <cursorShape>BusyCursor</cursorShape> | ||||
|   </property> | ||||
|   <property name="windowTitle"> | ||||
|    <string>UDS Connection</string> | ||||
|   </property> | ||||
|   <property name="windowIcon"> | ||||
|    <iconset resource="UDSResources.qrc"> | ||||
|     <normaloff>:/images/logo-uds-small</normaloff>:/images/logo-uds-small</iconset> | ||||
|   </property> | ||||
|   <property name="windowOpacity"> | ||||
|    <double>1.000000000000000</double> | ||||
|   </property> | ||||
|   <widget class="QWidget" name="centralwidget"> | ||||
|    <property name="autoFillBackground"> | ||||
|     <bool>true</bool> | ||||
|    </property> | ||||
|    <layout class="QVBoxLayout" name="verticalLayout_2"> | ||||
|     <property name="spacing"> | ||||
|      <number>4</number> | ||||
|     </property> | ||||
|     <property name="margin"> | ||||
|      <number>4</number> | ||||
|     </property> | ||||
|     <item> | ||||
|      <widget class="QFrame" name="frame"> | ||||
|       <property name="frameShape"> | ||||
|        <enum>QFrame::StyledPanel</enum> | ||||
|       </property> | ||||
|       <property name="frameShadow"> | ||||
|        <enum>QFrame::Raised</enum> | ||||
|       </property> | ||||
|       <layout class="QVBoxLayout" name="verticalLayout_3"> | ||||
|        <property name="spacing"> | ||||
|         <number>4</number> | ||||
|        </property> | ||||
|        <property name="margin"> | ||||
|         <number>4</number> | ||||
|        </property> | ||||
|        <item> | ||||
|         <layout class="QVBoxLayout" name="verticalLayout"> | ||||
|          <item> | ||||
|           <widget class="QLabel" name="image"> | ||||
|            <property name="minimumSize"> | ||||
|             <size> | ||||
|              <width>0</width> | ||||
|              <height>24</height> | ||||
|             </size> | ||||
|            </property> | ||||
|            <property name="autoFillBackground"> | ||||
|             <bool>true</bool> | ||||
|            </property> | ||||
|            <property name="text"> | ||||
|             <string/> | ||||
|            </property> | ||||
|            <property name="pixmap"> | ||||
|             <pixmap resource="UDSResources.qrc">:/images/logo-uds-small</pixmap> | ||||
|            </property> | ||||
|            <property name="scaledContents"> | ||||
|             <bool>false</bool> | ||||
|            </property> | ||||
|            <property name="alignment"> | ||||
|             <set>Qt::AlignCenter</set> | ||||
|            </property> | ||||
|           </widget> | ||||
|          </item> | ||||
|          <item> | ||||
|           <widget class="QLabel" name="info"> | ||||
|            <property name="maximumSize"> | ||||
|             <size> | ||||
|              <width>16777215</width> | ||||
|              <height>16</height> | ||||
|             </size> | ||||
|            </property> | ||||
|            <property name="text"> | ||||
|             <string>TextLabel</string> | ||||
|            </property> | ||||
|           </widget> | ||||
|          </item> | ||||
|          <item> | ||||
|           <widget class="QProgressBar" name="progressBar"> | ||||
|            <property name="value"> | ||||
|             <number>24</number> | ||||
|            </property> | ||||
|            <property name="textVisible"> | ||||
|             <bool>false</bool> | ||||
|            </property> | ||||
|           </widget> | ||||
|          </item> | ||||
|          <item> | ||||
|           <layout class="QHBoxLayout" name="horizontalLayout"> | ||||
|            <item> | ||||
|             <spacer name="horizontalSpacer"> | ||||
|              <property name="orientation"> | ||||
|               <enum>Qt::Horizontal</enum> | ||||
|              </property> | ||||
|              <property name="sizeHint" stdset="0"> | ||||
|               <size> | ||||
|                <width>40</width> | ||||
|                <height>20</height> | ||||
|               </size> | ||||
|              </property> | ||||
|             </spacer> | ||||
|            </item> | ||||
|            <item> | ||||
|             <widget class="QPushButton" name="cancelButton"> | ||||
|              <property name="text"> | ||||
|               <string>Cancel</string> | ||||
|              </property> | ||||
|              <property name="default"> | ||||
|               <bool>true</bool> | ||||
|              </property> | ||||
|              <property name="flat"> | ||||
|               <bool>false</bool> | ||||
|              </property> | ||||
|             </widget> | ||||
|            </item> | ||||
|            <item> | ||||
|             <spacer name="horizontalSpacer_2"> | ||||
|              <property name="orientation"> | ||||
|               <enum>Qt::Horizontal</enum> | ||||
|              </property> | ||||
|              <property name="sizeHint" stdset="0"> | ||||
|               <size> | ||||
|                <width>40</width> | ||||
|                <height>20</height> | ||||
|               </size> | ||||
|              </property> | ||||
|             </spacer> | ||||
|            </item> | ||||
|           </layout> | ||||
|          </item> | ||||
|         </layout> | ||||
|        </item> | ||||
|       </layout> | ||||
|      </widget> | ||||
|     </item> | ||||
|    </layout> | ||||
|   </widget> | ||||
|  </widget> | ||||
|  <resources> | ||||
|   <include location="UDSResources.qrc"/> | ||||
|  </resources> | ||||
|  <connections/> | ||||
| </ui> | ||||
							
								
								
									
										
											BIN
										
									
								
								client/full/src/images/logo-512.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 48 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/images/logo-uds-small.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.2 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/images/logo-uds.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 36 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/images/uds.ico
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 9.4 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/macosx/uds.icns
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								client/full/src/macosx/uds.iconset/icon_128x128.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 12 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/macosx/uds.iconset/icon_128x128@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 18 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/macosx/uds.iconset/icon_16x16.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 5.0 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/macosx/uds.iconset/icon_16x16@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 5.6 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/macosx/uds.iconset/icon_256x256.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 18 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/macosx/uds.iconset/icon_256x256@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 48 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/macosx/uds.iconset/icon_32x32.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 5.6 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/macosx/uds.iconset/icon_32x32@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 7.6 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/macosx/uds.iconset/icon_512x512.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 48 KiB | 
							
								
								
									
										
											BIN
										
									
								
								client/full/src/macosx/uds.iconset/icon_512x512@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 137 KiB | 
							
								
								
									
										44
									
								
								client/full/src/uds/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,44 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| # | ||||
| # Copyright (c) 2014 Virtual Cable S.L. | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without modification, | ||||
| # are permitted provided that the following conditions are met: | ||||
| # | ||||
| #    * Redistributions of source code must retain the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer. | ||||
| #    * Redistributions in binary form must reproduce the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer in the documentation | ||||
| #      and/or other materials provided with the distribution. | ||||
| #    * Neither the name of Virtual Cable S.L. nor the names of its contributors | ||||
| #      may be used to endorse or promote products derived from this software | ||||
| #      without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||
| # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | ||||
| # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | ||||
| # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | ||||
| # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
| # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
|  | ||||
| ''' | ||||
| @author: Adolfo Gómez, dkmaster at dkmon dot com | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| # On centos, old six release does not includes byte2int, nor six.PY2 | ||||
| import six | ||||
|  | ||||
| VERSION = '2.1.0' | ||||
|  | ||||
| __title__ = 'udclient' | ||||
| __version__ = VERSION | ||||
| __build__ = 0x010750 | ||||
| __author__ = 'Adolfo Gómez' | ||||
| __license__ = "BSD 3-clause" | ||||
| __copyright__ = "Copyright 2014-2015 VirtualCable S.L.U." | ||||
							
								
								
									
										193
									
								
								client/full/src/uds/forward.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,193 @@ | ||||
| # Based on forward.py from paramiko | ||||
| # Copyright (C) 2003-2007  Robey Pointer <robeypointer@gmail.com> | ||||
| # https://github.com/paramiko/paramiko/blob/master/demos/forward.py | ||||
|  | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import select | ||||
| import SocketServer | ||||
|  | ||||
| import paramiko | ||||
| import threading | ||||
| import random | ||||
| import time | ||||
|  | ||||
| from .log import logger | ||||
|  | ||||
| class ForwardServer(SocketServer.ThreadingTCPServer): | ||||
|     daemon_threads = True | ||||
|     allow_reuse_address = True | ||||
|  | ||||
|  | ||||
| class Handler(SocketServer.BaseRequestHandler): | ||||
|  | ||||
|     def handle(self): | ||||
|         self.thread.currentConnections += 1 | ||||
|  | ||||
|         try: | ||||
|             chan = self.ssh_transport.open_channel('direct-tcpip', | ||||
|                                                    (self.chain_host, self.chain_port), | ||||
|                                                    self.request.getpeername()) | ||||
|         except Exception as e: | ||||
|             logger.exception('Incoming request to %s:%d failed: %s' % (self.chain_host, | ||||
|                                                               self.chain_port, | ||||
|                                                               repr(e))) | ||||
|             return | ||||
|         if chan is None: | ||||
|             logger.error('Incoming request to %s:%d was rejected by the SSH server.' % | ||||
|                     (self.chain_host, self.chain_port)) | ||||
|             return | ||||
|  | ||||
|         logger.debug('Connected!  Tunnel open %r -> %r -> %r' % (self.request.getpeername(), | ||||
|                                                             chan.getpeername(), (self.chain_host, self.chain_port))) | ||||
|         try: | ||||
|             while self.event.is_set() is False: | ||||
|                 r, _w, _x = select.select([self.request, chan], [], [], 1) | ||||
|  | ||||
|                 if self.request in r: | ||||
|                     data = self.request.recv(1024) | ||||
|                     if len(data) == 0: | ||||
|                         break | ||||
|                     chan.send(data) | ||||
|                 if chan in r: | ||||
|                     data = chan.recv(1024) | ||||
|                     if len(data) == 0: | ||||
|                         break | ||||
|                     self.request.send(data) | ||||
|         except Exception: | ||||
|             pass | ||||
|  | ||||
|         try: | ||||
|             peername = self.request.getpeername() | ||||
|             chan.close() | ||||
|             self.request.close() | ||||
|             logger.debug('Tunnel closed from %r' % (peername,)) | ||||
|         except Exception: | ||||
|             pass | ||||
|  | ||||
|         self.thread.currentConnections -= 1 | ||||
|  | ||||
|         if self.thread.stoppable is True and self.thread.currentConnections == 0: | ||||
|             self.thread.stop() | ||||
|  | ||||
|  | ||||
| class ForwardThread(threading.Thread): | ||||
|     status = 0  # Connecting | ||||
|  | ||||
|     def __init__(self, server, port, username, password, localPort, redirectHost, redirectPort, waitTime): | ||||
|         threading.Thread.__init__(self) | ||||
|         self.client = None | ||||
|         self.fs = None | ||||
|  | ||||
|         self.server = server | ||||
|         self.port = int(port) | ||||
|         self.username = username | ||||
|         self.password = password | ||||
|  | ||||
|         self.localPort = int(localPort) | ||||
|         self.redirectHost = redirectHost | ||||
|         self.redirectPort = redirectPort | ||||
|  | ||||
|         self.waitTime = waitTime | ||||
|  | ||||
|         self.stopEvent = threading.Event() | ||||
|  | ||||
|         self.timer = None | ||||
|         self.currentConnections = 0 | ||||
|         self.stoppable = False | ||||
|         self.client = None | ||||
|  | ||||
|     def clone(self, redirectHost, redirectPort, localPort=None): | ||||
|         if localPort is None: | ||||
|             localPort = random.randrange(40000, 50000) | ||||
|  | ||||
|         ft = ForwardThread(self.server, self.port, self.username, self.password, localPort, redirectHost, redirectPort, self.waitTime) | ||||
|         ft.client = self.client | ||||
|         self.client.useCount += 1  # One more using this client | ||||
|         ft.start() | ||||
|  | ||||
|         while ft.status == 0: | ||||
|             time.sleep(0.1) | ||||
|  | ||||
|         return (ft, localPort) | ||||
|  | ||||
|  | ||||
|     def _timerFnc(self): | ||||
|         self.timer = None | ||||
|         logger.debug('Timer fnc: {}'.format(self.currentConnections)) | ||||
|         self.stoppable = True | ||||
|         if self.currentConnections <= 0: | ||||
|             self.stop() | ||||
|  | ||||
|     def run(self): | ||||
|         if self.client is None: | ||||
|             self.client = paramiko.SSHClient() | ||||
|             self.client.useCount = 1  # Custom added variable, to keep track on when to close tunnel | ||||
|             self.client.load_system_host_keys() | ||||
|             self.client.set_missing_host_key_policy(paramiko.AutoAddPolicy()) | ||||
|  | ||||
|             logger.debug('Connecting to ssh host %s:%d ...' % (self.server, self.port)) | ||||
|  | ||||
|             try: | ||||
|                 self.client.connect(self.server, self.port, username=self.username, password=self.password, timeout=5) | ||||
|             except Exception as e: | ||||
|                 logger.exception('Exception connecting: ') | ||||
|                 self.status = 2  # Error | ||||
|                 return | ||||
|  | ||||
|         class SubHandler(Handler): | ||||
|             chain_host = self.redirectHost | ||||
|             chain_port = self.redirectPort | ||||
|             ssh_transport = self.client.get_transport() | ||||
|             event = self.stopEvent | ||||
|             thread = self | ||||
|  | ||||
|         logger.debug('Wait Time: {}'.format(self.waitTime)) | ||||
|         self.timer = threading.Timer(self.waitTime, self._timerFnc) | ||||
|         self.timer.start() | ||||
|  | ||||
|         self.status = 1  # Ok, listening | ||||
|  | ||||
|         self.fs = ForwardServer(('', self.localPort), SubHandler) | ||||
|         self.fs.serve_forever() | ||||
|  | ||||
|     def stop(self): | ||||
|         try: | ||||
|             if self.timer: | ||||
|                 self.timer.cancel() | ||||
|  | ||||
|             self.stopEvent.set() | ||||
|             self.fs.shutdown() | ||||
|  | ||||
|             if self.client is not None: | ||||
|                 self.client.useCount -= 1 | ||||
|                 if self.client.useCount == 0: | ||||
|                     self.client.close() | ||||
|                 self.client = None  # Clean up | ||||
|         except Exception: | ||||
|             logger.exception('Exception stopping') | ||||
|             pass | ||||
|  | ||||
|  | ||||
| def forward(server, port, username, password, redirectHost, redirectPort, localPort=None, waitTime=10): | ||||
|     ''' | ||||
|     Instantiates an ssh connection to server:port | ||||
|     Returns the Thread created and the local redirected port as a list: (thread, port) | ||||
|     ''' | ||||
|     port, redirectPort = int(port), int(redirectPort) | ||||
|  | ||||
|     if localPort is None: | ||||
|         localPort = random.randrange(40000, 50000) | ||||
|  | ||||
|     logger.debug('Connecting to {}:{} using {}/{} redirecting to {}:{}, listening on 127.0.0.1:{}'.format( | ||||
|         server, port, username, password, redirectHost, redirectPort, localPort)) | ||||
|  | ||||
|     ft = ForwardThread(server, port, username, password, localPort, redirectHost, redirectPort, waitTime) | ||||
|  | ||||
|     ft.start() | ||||
|  | ||||
|     while ft.status == 0: | ||||
|         time.sleep(0.1) | ||||
|  | ||||
|     return (ft, localPort) | ||||
|  | ||||
							
								
								
									
										52
									
								
								client/full/src/uds/log.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,52 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
| # | ||||
| # Copyright (c) 2014 Virtual Cable S.L. | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without modification, | ||||
| # are permitted provided that the following conditions are met: | ||||
| # | ||||
| #    * Redistributions of source code must retain the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer. | ||||
| #    * Redistributions in binary form must reproduce the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer in the documentation | ||||
| #      and/or other materials provided with the distribution. | ||||
| #    * Neither the name of Virtual Cable S.L. nor the names of its contributors | ||||
| #      may be used to endorse or promote products derived from this software | ||||
| #      without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||
| # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | ||||
| # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | ||||
| # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | ||||
| # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
| # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
|  | ||||
| ''' | ||||
| @author: Adolfo Gómez, dkmaster at dkmon dot com | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import logging | ||||
| import os | ||||
| import sys | ||||
| import tempfile | ||||
|  | ||||
| if sys.platform.startswith('linux'): | ||||
|     from os.path import expanduser | ||||
|     logFile = expanduser('~/udsclient.log') | ||||
| else: | ||||
|     logFile = os.path.join(tempfile.gettempdir(), b'udsclient.log') | ||||
|  | ||||
| logging.basicConfig( | ||||
|     filename=logFile, | ||||
|     filemode='a', | ||||
|     format='%(levelname)s %(asctime)s %(message)s', | ||||
|     level=logging.DEBUG | ||||
| ) | ||||
|  | ||||
| logger = logging.getLogger('udsclient') | ||||
							
								
								
									
										49
									
								
								client/full/src/uds/osDetector.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,49 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # | ||||
| # Copyright (c) 2017 Virtual Cable S.L. | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without modification, | ||||
| # are permitted provided that the following conditions are met: | ||||
| # | ||||
| #    * Redistributions of source code must retain the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer. | ||||
| #    * Redistributions in binary form must reproduce the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer in the documentation | ||||
| #      and/or other materials provided with the distribution. | ||||
| #    * Neither the name of Virtual Cable S.L. nor the names of its contributors | ||||
| #      may be used to endorse or promote products derived from this software | ||||
| #      without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||
| # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | ||||
| # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | ||||
| # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | ||||
| # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
| # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
|  | ||||
| ''' | ||||
| @author: Adolfo Gómez, dkmaster at dkmon dot com | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import sys | ||||
|  | ||||
|  | ||||
| LINUX = 'Linux' | ||||
| WINDOWS = 'Windows' | ||||
| MAC_OS_X = 'Mac os x' | ||||
|  | ||||
|  | ||||
| def getOs(): | ||||
|     if sys.platform.startswith('linux'): | ||||
|         return LINUX | ||||
|     elif sys.platform.startswith('win'): | ||||
|         return WINDOWS | ||||
|     elif sys.platform.startswith('darwin'): | ||||
|         return MAC_OS_X | ||||
							
								
								
									
										116
									
								
								client/full/src/uds/rest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,116 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # | ||||
| # Copyright (c) 2017 Virtual Cable S.L. | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without modification, | ||||
| # are permitted provided that the following conditions are met: | ||||
| # | ||||
| #    * Redistributions of source code must retain the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer. | ||||
| #    * Redistributions in binary form must reproduce the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer in the documentation | ||||
| #      and/or other materials provided with the distribution. | ||||
| #    * Neither the name of Virtual Cable S.L. nor the names of its contributors | ||||
| #      may be used to endorse or promote products derived from this software | ||||
| #      without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||
| # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | ||||
| # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | ||||
| # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | ||||
| # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
| # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
|  | ||||
| ''' | ||||
| @author: Adolfo Gómez, dkmaster at dkmon dot com | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from PyQt4.QtCore import pyqtSignal, pyqtSlot | ||||
| from PyQt4.QtCore import QObject, QUrl, QSettings | ||||
| from PyQt4.QtCore import Qt | ||||
| from PyQt4.QtNetwork import QNetworkAccessManager, QNetworkRequest, QNetworkReply, QSslCertificate | ||||
| from PyQt4.QtGui import QMessageBox | ||||
| from . import VERSION | ||||
|  | ||||
| import json | ||||
| import osDetector | ||||
| import six | ||||
| import urllib | ||||
|  | ||||
|  | ||||
| class RestRequest(QObject): | ||||
|  | ||||
|     restApiUrl = ''  # | ||||
|  | ||||
|     done = pyqtSignal(dict, name='done') | ||||
|  | ||||
|     def __init__(self, url, parentWindow, done, params=None):  # parent not used | ||||
|         super(RestRequest, self).__init__() | ||||
|         # private | ||||
|         self._manager = QNetworkAccessManager() | ||||
|         if params is not None: | ||||
|             url += '?' + '&'.join('{}={}'.format(k, urllib.quote(six.text_type(v).encode('utf8'))) for k, v in params.iteritems()) | ||||
|  | ||||
|         self.url = QUrl(RestRequest.restApiUrl + url) | ||||
|  | ||||
|         # connect asynchronous result, when a request finishes | ||||
|         self._manager.finished.connect(self._finished) | ||||
|         self._manager.sslErrors.connect(self._sslError) | ||||
|         self._parentWindow = parentWindow | ||||
|  | ||||
|         self.done.connect(done, Qt.QueuedConnection) | ||||
|  | ||||
|     # private slot, no need to declare as slot | ||||
|     @pyqtSlot(QNetworkReply) | ||||
|     def _finished(self, reply): | ||||
|         ''' | ||||
|         Handle signal 'finished'.  A network request has finished. | ||||
|         ''' | ||||
|         try: | ||||
|             if reply.error() != QNetworkReply.NoError: | ||||
|                 raise Exception(reply.errorString()) | ||||
|             data = six.text_type(reply.readAll()) | ||||
|             data = json.loads(data) | ||||
|         except Exception as e: | ||||
|             data = { | ||||
|                 'result': None, | ||||
|                 'error': six.text_type(e) | ||||
|             } | ||||
|  | ||||
|         self.done.emit(data) | ||||
|  | ||||
|         reply.deleteLater()  # schedule for delete from main event loop | ||||
|  | ||||
|     @pyqtSlot(QNetworkReply, list) | ||||
|     def _sslError(self, reply, errors): | ||||
|         settings = QSettings() | ||||
|         settings.beginGroup('ssl') | ||||
|         cert = errors[0].certificate() | ||||
|         digest = six.text_type(cert.digest().toHex()) | ||||
|  | ||||
|         approved = settings.value(digest, False).toBool() | ||||
|  | ||||
|         errorString = '<p>The certificate for <b>{}</b> has the following errors:</p><ul>'.format(cert.subjectInfo(QSslCertificate.CommonName)) | ||||
|  | ||||
|         for err in errors: | ||||
|             errorString += '<li>' + err.errorString() + '</li>' | ||||
|  | ||||
|         errorString += '</ul>' | ||||
|  | ||||
|         if approved or QMessageBox.warning(self._parentWindow, 'SSL Warning', errorString, QMessageBox.Yes | QMessageBox.No) == QMessageBox.Yes: | ||||
|             settings.setValue(digest, True) | ||||
|             reply.ignoreSslErrors() | ||||
|  | ||||
|         settings.endGroup() | ||||
|  | ||||
|     def get(self): | ||||
|         request = QNetworkRequest(self.url) | ||||
|         request.setRawHeader('User-Agent', osDetector.getOs() + " - UDS Connector " + VERSION) | ||||
|         self._manager.get(request) | ||||
							
								
								
									
										162
									
								
								client/full/src/uds/tools.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,162 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # | ||||
| # Copyright (c) 2015 Virtual Cable S.L. | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without modification, | ||||
| # are permitted provided that the following conditions are met: | ||||
| # | ||||
| #    * Redistributions of source code must retain the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer. | ||||
| #    * Redistributions in binary form must reproduce the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer in the documentation | ||||
| #      and/or other materials provided with the distribution. | ||||
| #    * Neither the name of Virtual Cable S.L. nor the names of its contributors | ||||
| #      may be used to endorse or promote products derived from this software | ||||
| #      without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||
| # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | ||||
| # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | ||||
| # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | ||||
| # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
| # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
|  | ||||
| ''' | ||||
| @author: Adolfo Gómez, dkmaster at dkmon dot com | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import tempfile | ||||
| import string | ||||
| import random | ||||
| import os | ||||
| import socket | ||||
| import stat | ||||
| import six | ||||
| import sys | ||||
| import time | ||||
|  | ||||
| from log import logger | ||||
|  | ||||
| _unlinkFiles = [] | ||||
| _tasksToWait = [] | ||||
| _execBeforeExit = [] | ||||
|  | ||||
|  | ||||
| sys_fs_enc = sys.getfilesystemencoding() or 'mbcs' | ||||
|  | ||||
| def saveTempFile(content, filename=None): | ||||
|     if filename is None: | ||||
|         filename = b''.join(random.choice(string.ascii_lowercase + string.digits) for _ in range(16)) | ||||
|         filename = filename + '.uds' | ||||
|  | ||||
|     if 'win32' in sys.platform: | ||||
|         logger.info('Fixing for win32') | ||||
|         filename = filename.encode(sys_fs_enc) | ||||
|  | ||||
|     filename = os.path.join(tempfile.gettempdir(), filename) | ||||
|  | ||||
|     with open(filename, 'w') as f: | ||||
|         f.write(content) | ||||
|  | ||||
|     logger.info('Returning filename') | ||||
|     return filename | ||||
|  | ||||
| def readTempFile(filename): | ||||
|     if 'win32' in sys.platform: | ||||
|         filename = filename.encode('utf-8') | ||||
|  | ||||
|     filename = os.path.join(tempfile.gettempdir(), filename) | ||||
|     try: | ||||
|         with open(filename, 'r') as f: | ||||
|             return f.read() | ||||
|     except Exception: | ||||
|         return None | ||||
|  | ||||
| def testServer(host, port, timeOut=4): | ||||
|     try: | ||||
|         sock = socket.create_connection((host, int(port)), timeOut) | ||||
|         sock.close() | ||||
|     except Exception: | ||||
|         return False | ||||
|     return True | ||||
|  | ||||
|  | ||||
| def findApp(appName, extraPath=None): | ||||
|     if 'win32' in sys.platform and isinstance(appName, six.text_type): | ||||
|         appName = appName.encode(sys_fs_enc) | ||||
|     searchPath = os.environ['PATH'].split(os.pathsep) | ||||
|     if extraPath is not None: | ||||
|         searchPath += list(extraPath) | ||||
|  | ||||
|     for path in searchPath: | ||||
|         fileName = os.path.join(path, appName) | ||||
|         if os.path.isfile(fileName) and (os.stat(fileName).st_mode & stat.S_IXUSR) != 0: | ||||
|             return fileName | ||||
|     return None | ||||
|  | ||||
|  | ||||
| def getHostName(): | ||||
|     ''' | ||||
|     Returns current host name | ||||
|     In fact, it's a wrapper for socket.gethostname() | ||||
|     ''' | ||||
|     hostname = socket.gethostname() | ||||
|     if 'win32' in sys.platform: | ||||
|         hostname = hostname.decode(sys_fs_enc) | ||||
|  | ||||
|     hostname = six.text_type(hostname) | ||||
|     logger.info('Hostname: {}'.format(hostname)) | ||||
|     return hostname | ||||
|  | ||||
| # Queing operations (to be executed before exit) | ||||
|  | ||||
|  | ||||
| def addFileToUnlink(filename): | ||||
|     ''' | ||||
|     Adds a file to the wait-and-unlink list | ||||
|     ''' | ||||
|     _unlinkFiles.append(filename) | ||||
|  | ||||
|  | ||||
| def unlinkFiles(): | ||||
|     ''' | ||||
|     Removes all wait-and-unlink files | ||||
|     ''' | ||||
|     if len(_unlinkFiles) > 0: | ||||
|         time.sleep(5)  # Wait 5 seconds before deleting anything | ||||
|     for f in _unlinkFiles: | ||||
|         try: | ||||
|             os.unlink(f) | ||||
|         except Exception: | ||||
|             pass | ||||
|  | ||||
|  | ||||
| def addTaskToWait(taks): | ||||
|     _tasksToWait.append(taks) | ||||
|  | ||||
|  | ||||
| def waitForTasks(): | ||||
|     for t in _tasksToWait: | ||||
|         try: | ||||
|             if hasattr(t, 'join'): | ||||
|                 t.join() | ||||
|             elif hasattr(t, 'wait'): | ||||
|                 t.wait() | ||||
|         except Exception: | ||||
|             pass | ||||
|  | ||||
|  | ||||
| def addExecBeforeExit(fnc): | ||||
|     _execBeforeExit.append(fnc) | ||||
|  | ||||
|  | ||||
| def execBeforeExit(): | ||||
|     for fnc in _execBeforeExit: | ||||
|         fnc.__call__() | ||||
							
								
								
									
										15
									
								
								client/full/src/update.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
|  | ||||
| function process { | ||||
|     for a in *.ui; do | ||||
|         pyuic4 $a -o `basename $a .ui`.py -x | ||||
|     done | ||||
| }     | ||||
|  | ||||
| pyrcc4 -py3 UDSResources.qrc -o UDSResources_rc.py | ||||
|  | ||||
|  | ||||
| # process current directory ui's | ||||
| process | ||||
|  | ||||
| @@ -1,17 +1,17 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <projectDescription> | ||||
| 	<name>tunnelLaucher</name> | ||||
| 	<name>UDSClient-Thin</name> | ||||
| 	<comment></comment> | ||||
| 	<projects> | ||||
| 	</projects> | ||||
| 	<buildSpec> | ||||
| 		<buildCommand> | ||||
| 			<name>org.eclipse.jdt.core.javabuilder</name> | ||||
| 			<name>org.python.pydev.PyDevBuilder</name> | ||||
| 			<arguments> | ||||
| 			</arguments> | ||||
| 		</buildCommand> | ||||
| 	</buildSpec> | ||||
| 	<natures> | ||||
| 		<nature>org.eclipse.jdt.core.javanature</nature> | ||||
| 		<nature>org.python.pydev.pythonNature</nature> | ||||
| 	</natures> | ||||
| </projectDescription> | ||||
| @@ -1,8 +1,8 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <?eclipse-pydev version="1.0"?><pydev_project> | ||||
| <pydev_property name="org.python.pydev.PYTHON_PROJECT_INTERPRETER">Default</pydev_property> | ||||
| <pydev_property name="org.python.pydev.PYTHON_PROJECT_VERSION">python 2.6</pydev_property> | ||||
| <pydev_pathproperty name="org.python.pydev.PROJECT_SOURCE_PATH"> | ||||
| <path>/linuxActor/src</path> | ||||
| <path>/${PROJECT_DIR_NAME}/src</path> | ||||
| </pydev_pathproperty> | ||||
| <pydev_property name="org.python.pydev.PYTHON_PROJECT_VERSION">python 2.7</pydev_property> | ||||
| <pydev_property name="org.python.pydev.PYTHON_PROJECT_INTERPRETER">Default</pydev_property> | ||||
| </pydev_project> | ||||
							
								
								
									
										158
									
								
								client/thin/src/UDSClient.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,158 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # | ||||
| # Copyright (c) 2017 Virtual Cable S.L. | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without modification, | ||||
| # are permitted provided that the following conditions are met: | ||||
| # | ||||
| #    * Redistributions of source code must retain the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer. | ||||
| #    * Redistributions in binary form must reproduce the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer in the documentation | ||||
| #      and/or other materials provided with the distribution. | ||||
| #    * Neither the name of Virtual Cable S.L. nor the names of its contributors | ||||
| #      may be used to endorse or promote products derived from this software | ||||
| #      without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||
| # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | ||||
| # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | ||||
| # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | ||||
| # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
| # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
| ''' | ||||
| @author: Adolfo Gómez, dkmaster at dkmon dot com | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from uds import ui | ||||
| from uds.rest import RestRequest, RetryException | ||||
| from uds.forward import forward | ||||
| from uds import VERSION | ||||
| from uds.log import logger  # @UnresolvedImport | ||||
| from uds import tools | ||||
|  | ||||
| import six | ||||
| import sys | ||||
| import pickle | ||||
|  | ||||
|  | ||||
| def approveHost(host): | ||||
|     from os.path import expanduser | ||||
|     hostsFile = expanduser('~/.udsclient.hosts') | ||||
|  | ||||
|     try: | ||||
|         with open(hostsFile, 'r') as f: | ||||
|             approvedHosts = f.read().splitlines() | ||||
|     except Exception: | ||||
|         approvedHosts = [] | ||||
|  | ||||
|     host = host.lower() | ||||
|  | ||||
|     if host in approvedHosts: | ||||
|         return True | ||||
|  | ||||
|     errorString = 'The server {} must be approved:\n'.format(host) | ||||
|     errorString += 'Only approve UDS servers that you trust to avoid security issues.' | ||||
|  | ||||
|     approved = ui.question("ACCESS Warning", errorString) | ||||
|  | ||||
|     if approved: | ||||
|         approvedHosts.append(host) | ||||
|         logger.debug('Host was approved, saving to approvedHosts file') | ||||
|         try: | ||||
|             with open(hostsFile, 'w') as f: | ||||
|                 f.write('\n'.join(approvedHosts)) | ||||
|         except Exception: | ||||
|             logger.warn('Got exception writing to {}'.format(hostsFile)) | ||||
|  | ||||
|     return approved | ||||
|  | ||||
|  | ||||
| def getWithRetry(rest, url, params=None): | ||||
|     while True: | ||||
|         try: | ||||
|             res = rest.get(url, params) | ||||
|             return res | ||||
|         except RetryException as e: | ||||
|             if ui.question('Service not available', 'Error {}.\nPlease, wait a minute and press "OK" to retry, or "CANCEL" to abort'.format(e)) is True: | ||||
|                 continue | ||||
|             raise Exception('Cancelled by user') | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     logger.debug('Initializing connector') | ||||
|  | ||||
|     if six.PY3 is False: | ||||
|         logger.debug('Fixing threaded execution of commands') | ||||
|         import threading | ||||
|         threading._DummyThread._Thread__stop = lambda x: 42 | ||||
|  | ||||
|     # First parameter must be url | ||||
|     try: | ||||
|         uri = sys.argv[1] | ||||
|  | ||||
|         if uri == '--test': | ||||
|             sys.exit(0) | ||||
|  | ||||
|         logger.debug('URI: {}'.format(uri)) | ||||
|         if uri[:6] != 'uds://' and uri[:7] != 'udss://': | ||||
|             raise Exception() | ||||
|  | ||||
|         ssl = uri[3] == 's' | ||||
|         host, ticket, scrambler = uri.split('//')[1].split('/') | ||||
|         logger.debug('ssl: {}, host:{}, ticket:{}, scrambler:{}'.format(ssl, host, ticket, scrambler)) | ||||
|  | ||||
|     except Exception: | ||||
|         logger.debug('Detected execution without valid URI, exiting') | ||||
|         ui.message('UDS Client', 'UDS Client Version {}'.format(VERSION)) | ||||
|         sys.exit(1) | ||||
|  | ||||
|     rest = RestRequest(host, ssl) | ||||
|     logger.debug('Setting request URL to {}'.format(rest.restApiUrl)) | ||||
|  | ||||
|     # Main requests part | ||||
|     # First, get version | ||||
|     try: | ||||
|         res = getWithRetry(rest, '') | ||||
|  | ||||
|         logger.debug('Got information {}'.format(res)) | ||||
|  | ||||
|         if res['requiredVersion'] > VERSION: | ||||
|             ui.message("New UDS Client available", "A new uds version is needed in order to access this version of UDS.\nPlease, download and install it") | ||||
|             sys.exit(1) | ||||
|  | ||||
|         res = getWithRetry(rest, '/{}/{}'.format(ticket, scrambler), params={'hostname': tools.getHostName(), 'version': VERSION}) | ||||
|  | ||||
|         script = res.decode('base64').decode('bz2') | ||||
|  | ||||
|         logger.debug('Script: {}'.format(script)) | ||||
|  | ||||
|         six.exec_(script, globals(), {'parent': None}) | ||||
|     except Exception as e: | ||||
|         error = 'ERROR: {}'.format(e) | ||||
|         logger.error(error) | ||||
|         ui.message('Error', error) | ||||
|         sys.exit(2) | ||||
|  | ||||
|     # Finalize | ||||
|     try: | ||||
|         tools.waitForTasks() | ||||
|     except Exception: | ||||
|         pass | ||||
|  | ||||
|     try: | ||||
|         tools.unlinkFiles() | ||||
|     except Exception: | ||||
|         pass | ||||
|  | ||||
|     try: | ||||
|         tools.execBeforeExit() | ||||
|     except Exception: | ||||
|         pass | ||||
							
								
								
									
										1
									
								
								client/thin/src/uds/__init__.py
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| ../../../full/src/uds/__init__.py | ||||
							
								
								
									
										1
									
								
								client/thin/src/uds/forward.py
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| ../../../full/src/uds/forward.py | ||||
							
								
								
									
										1
									
								
								client/thin/src/uds/log.py
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| ../../../full/src/uds/log.py | ||||
							
								
								
									
										1
									
								
								client/thin/src/uds/osDetector.py
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| ../../../full/src/uds/osDetector.py | ||||
							
								
								
									
										86
									
								
								client/thin/src/uds/rest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,86 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # | ||||
| # Copyright (c) 2015 Virtual Cable S.L. | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without modification, | ||||
| # are permitted provided that the following conditions are met: | ||||
| # | ||||
| #    * Redistributions of source code must retain the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer. | ||||
| #    * Redistributions in binary form must reproduce the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer in the documentation | ||||
| #      and/or other materials provided with the distribution. | ||||
| #    * Neither the name of Virtual Cable S.L. nor the names of its contributors | ||||
| #      may be used to endorse or promote products derived from this software | ||||
| #      without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||
| # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | ||||
| # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | ||||
| # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | ||||
| # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
| # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
|  | ||||
| ''' | ||||
| @author: Adolfo Gómez, dkmaster at dkmon dot com | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import requests | ||||
| from . import VERSION | ||||
|  | ||||
| import json | ||||
| import six | ||||
| import osDetector | ||||
|  | ||||
|  | ||||
| from .log import logger | ||||
|  | ||||
| class RetryException(Exception): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class RestRequest(object): | ||||
|  | ||||
|     restApiUrl = '' | ||||
|  | ||||
|     def __init__(self, host, ssl=True):  # parent not used | ||||
|         super(RestRequest, self).__init__() | ||||
|  | ||||
|         self.host = host | ||||
|         self.ssl = ssl | ||||
|         self.restApiUrl = '{}://{}/rest/client'.format(['http', 'https'][ssl], host) | ||||
|  | ||||
|     def get(self, url, params=None): | ||||
|         url = self.restApiUrl + url | ||||
|         if params is not None: | ||||
|             url += '?' + '&'.join('{}={}'.format(k, six.moves.urllib.parse.quote(six.text_type(v).encode('utf8'))) for k, v in params.iteritems())  # @UndefinedVariable | ||||
|  | ||||
|         logger.debug('Requesting {}'.format(url)) | ||||
|  | ||||
|         try: | ||||
|             r = requests.get(url, headers={'Content-type': 'application/json', 'User-Agent': osDetector.getOs() + " - UDS Connector " + VERSION }) | ||||
|         except requests.exceptions.ConnectionError as e: | ||||
|             raise Exception('Error connecting to UDS Server at {}'.format(self.restApiUrl[0:-11])) | ||||
|  | ||||
|         if r.ok: | ||||
|             logger.debug('Request was OK. {}'.format(r.text)) | ||||
|             data = json.loads(r.text) | ||||
|             if not 'error' in data: | ||||
|                 return data['result'] | ||||
|             # Has error | ||||
|             if data.get('retryable', '0') == '1': | ||||
|                 raise RetryException(data['error']) | ||||
|  | ||||
|             raise Exception(data['error']) | ||||
|         else: | ||||
|             logger.error('Error requesting {}: {}, {}'.format(url, r.code. r.text)) | ||||
|             raise Exception('Error {}: {}'.format(r.code, r.text)) | ||||
|  | ||||
|         return data | ||||
							
								
								
									
										1
									
								
								client/thin/src/uds/tools.py
									
									
									
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| ../../../full/src/uds/tools.py | ||||
							
								
								
									
										44
									
								
								client/thin/src/uds/ui/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,44 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # | ||||
| # Copyright (c) 2017 Virtual Cable S.L. | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without modification, | ||||
| # are permitted provided that the following conditions are met: | ||||
| # | ||||
| #    * Redistributions of source code must retain the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer. | ||||
| #    * Redistributions in binary form must reproduce the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer in the documentation | ||||
| #      and/or other materials provided with the distribution. | ||||
| #    * Neither the name of Virtual Cable S.L. nor the names of its contributors | ||||
| #      may be used to endorse or promote products derived from this software | ||||
| #      without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||
| # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | ||||
| # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | ||||
| # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | ||||
| # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
| # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
| ''' | ||||
| @author: Adolfo Gómez, dkmaster at dkmon dot com | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| try: | ||||
|     import gtkui as theUI | ||||
| except Exception: | ||||
|     import consoleui as theUI  # @Reimport | ||||
|  | ||||
| def message(title, message): | ||||
|     theUI.message(title, message) | ||||
|  | ||||
| def question(title, message): | ||||
|     return theUI.question(title, message) | ||||
|  | ||||
							
								
								
									
										58
									
								
								client/thin/src/uds/ui/browser.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,58 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # | ||||
| # Copyright (c) 2017 Virtual Cable S.L. | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without modification, | ||||
| # are permitted provided that the following conditions are met: | ||||
| # | ||||
| #    * Redistributions of source code must retain the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer. | ||||
| #    * Redistributions in binary form must reproduce the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer in the documentation | ||||
| #      and/or other materials provided with the distribution. | ||||
| #    * Neither the name of Virtual Cable S.L. nor the names of its contributors | ||||
| #      may be used to endorse or promote products derived from this software | ||||
| #      without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||
| # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | ||||
| # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | ||||
| # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | ||||
| # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
| # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
| ''' | ||||
| @author: Adolfo Gómez, dkmaster at dkmon dot com | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
| import random | ||||
| import os | ||||
| import tempfile | ||||
| import string | ||||
| import webbrowser | ||||
|  | ||||
| TEMPLATE = '''<html> | ||||
|   <head> | ||||
|     <title>{title}</title> | ||||
|   </head> | ||||
|   <body> | ||||
|     <h1>{title}</h1> | ||||
|     <p>{message}<P> | ||||
|   </body> | ||||
| </html> | ||||
| ''' | ||||
|  | ||||
| def _htmlFilename(): | ||||
|     return os.path.join(tempfile.gettempdir(), ''.join([random.choice(string.ascii_lowercase) for i in range(22)]) + '.html') | ||||
|  | ||||
| def message(title, message): | ||||
|     filename = _htmlFilename() | ||||
|     with open(filename, 'w') as f: | ||||
|         f.write(TEMPLATE.format(title=title, message=message)) | ||||
|  | ||||
|     webbrowser.open('file://' + filename, new=0, autoraise=False) | ||||
							
								
								
									
										49
									
								
								client/thin/src/uds/ui/consoleui.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,49 @@ | ||||
| # -*- coding: utf-8 -*- | ||||
|  | ||||
| # | ||||
| # Copyright (c) 2017 Virtual Cable S.L. | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without modification, | ||||
| # are permitted provided that the following conditions are met: | ||||
| # | ||||
| #    * Redistributions of source code must retain the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer. | ||||
| #    * Redistributions in binary form must reproduce the above copyright notice, | ||||
| #      this list of conditions and the following disclaimer in the documentation | ||||
| #      and/or other materials provided with the distribution. | ||||
| #    * Neither the name of Virtual Cable S.L. nor the names of its contributors | ||||
| #      may be used to endorse or promote products derived from this software | ||||
| #      without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||||
| # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | ||||
| # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | ||||
| # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | ||||
| # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | ||||
| # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
| ''' | ||||
| @author: Adolfo Gómez, dkmaster at dkmon dot com | ||||
| ''' | ||||
| from __future__ import unicode_literals | ||||
| import sys | ||||
| import time | ||||
| from uds.log import logger | ||||
|  | ||||
| counter = 0 | ||||
|  | ||||
| def message(title, message): | ||||
|     sys.stderr.write("** {} **\n {}\n".format(title, message)) | ||||
|  | ||||
| def question(title, message): | ||||
|     global counter | ||||
|     if counter > 100:  # 5 minutes | ||||
|         return False | ||||
|     counter += 1 | ||||
|     sys.stderr.write("** {} **\n{}\nReturning YES in 3 seconds. (counter is {})\n".format(title, message, counter)) | ||||
|     time.sleep(3)  # Wait 3 seconds before returning | ||||
|     return True | ||||